Diff to HTML by rtfpessoa

Files changed (15) hide show
  1. /home/runner/work/progress/progress/clones/beta/{beta/custom_components → ha-core/homeassistant/components}/plugwise/__init__.py +41 -85
  2. /home/runner/work/progress/progress/clones/beta/{beta/custom_components → ha-core/homeassistant/components}/plugwise/binary_sensor.py +98 -68
  3. /home/runner/work/progress/progress/clones/beta/{beta/custom_components → ha-core/homeassistant/components}/plugwise/climate.py +127 -106
  4. /home/runner/work/progress/progress/clones/beta/{beta/custom_components → ha-core/homeassistant/components}/plugwise/config_flow.py +135 -180
  5. /home/runner/work/progress/progress/clones/beta/{beta/custom_components → ha-core/homeassistant/components}/plugwise/const.py +47 -34
  6. /home/runner/work/progress/progress/clones/beta/{beta/custom_components → ha-core/homeassistant/components}/plugwise/coordinator.py +98 -67
  7. /home/runner/work/progress/progress/clones/beta/{beta/custom_components → ha-core/homeassistant/components}/plugwise/diagnostics.py +5 -14
  8. /home/runner/work/progress/progress/clones/beta/{beta/custom_components → ha-core/homeassistant/components}/plugwise/entity.py +12 -14
  9. /home/runner/work/progress/progress/clones/beta/{beta/custom_components → ha-core/homeassistant/components}/plugwise/manifest.json +5 -5
  10. /home/runner/work/progress/progress/clones/beta/{beta/custom_components → ha-core/homeassistant/components}/plugwise/number.py +49 -77
  11. /home/runner/work/progress/progress/clones/beta/{beta/custom_components → ha-core/homeassistant/components}/plugwise/select.py +58 -67
  12. /home/runner/work/progress/progress/clones/beta/{beta/custom_components → ha-core/homeassistant/components}/plugwise/sensor.py +415 -32
  13. /home/runner/work/progress/progress/clones/beta/{beta/custom_components → ha-core/homeassistant/components}/plugwise/strings.json +107 -72
  14. /home/runner/work/progress/progress/clones/beta/{beta/custom_components → ha-core/homeassistant/components}/plugwise/switch.py +67 -26
  15. /home/runner/work/progress/progress/clones/beta/{beta/custom_components → ha-core/homeassistant/components}/plugwise/util.py +13 -12
/home/runner/work/progress/progress/clones/beta/{beta/custom_components → ha-core/homeassistant/components}/plugwise/__init__.py RENAMED
@@ -1,54 +1,26 @@
1
  """Plugwise platform for Home Assistant Core."""
 
2
  from __future__ import annotations
3
 
4
  from typing import Any
5
 
6
- import voluptuous as vol
7
-
8
- from homeassistant.components.number import DOMAIN as NUMBER_DOMAIN
9
- from homeassistant.components.switch import DOMAIN as SWITCH_DOMAIN
10
- from homeassistant.config_entries import ConfigEntry
11
  from homeassistant.const import Platform
12
- from homeassistant.core import HomeAssistant, ServiceCall, callback
13
  from homeassistant.helpers import device_registry as dr, entity_registry as er
14
- from plugwise.exceptions import PlugwiseError
15
 
16
- from .const import (
17
- CONF_REFRESH_INTERVAL, # pw-beta options
18
- COORDINATOR,
19
- DOMAIN,
20
- LOGGER,
21
- PLATFORMS_GATEWAY,
22
- SERVICE_DELETE,
23
- UNDO_UPDATE_LISTENER,
24
- )
25
- from .coordinator import PlugwiseDataUpdateCoordinator
26
 
27
 
28
- async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
29
- """Set up Plugwise Smiles from a config entry."""
30
  await er.async_migrate_entries(hass, entry.entry_id, async_migrate_entity_entry)
31
 
32
- cooldown = 1.5 # pw-beta frontend refresh-interval
33
- if (
34
- custom_refresh := entry.options.get(CONF_REFRESH_INTERVAL)
35
- ) is not None: # pragma: no cover
36
- cooldown = custom_refresh
37
- LOGGER.debug("DUC cooldown interval: %s", cooldown)
38
-
39
- coordinator = PlugwiseDataUpdateCoordinator(
40
- hass, entry, cooldown
41
- ) # pw-beta - cooldown, update_interval as extra
42
  await coordinator.async_config_entry_first_refresh()
43
- # Migrate a changed sensor unique_id
44
  migrate_sensor_entities(hass, coordinator)
45
 
46
- undo_listener = entry.add_update_listener(_update_listener) # pw-beta
47
-
48
- hass.data.setdefault(DOMAIN, {})[entry.entry_id] = {
49
- COORDINATOR: coordinator, # pw-beta
50
- UNDO_UPDATE_LISTENER: undo_listener, # pw-beta
51
- }
52
 
53
  device_registry = dr.async_get(hass)
54
  device_registry.async_get_or_create(
@@ -56,66 +28,44 @@
56
  identifiers={(DOMAIN, str(coordinator.api.gateway_id))},
57
  manufacturer="Plugwise",
58
  model=coordinator.api.smile_model,
 
59
  name=coordinator.api.smile_name,
60
- sw_version=coordinator.api.smile_version[0],
61
- )
62
 
63
- async def delete_notification(
64
- call: ServiceCall,
65
- ) -> None: # pragma: no cover # pw-beta: HA service - delete_notification
66
- """Service: delete the Plugwise Notification."""
67
- LOGGER.debug(
68
- "Service delete PW Notification called for %s",
69
- coordinator.api.smile_name,
70
- )
71
- try:
72
- await coordinator.api.delete_notification()
73
- LOGGER.debug("PW Notification deleted")
74
- except PlugwiseError:
75
- LOGGER.debug(
76
- "Failed to delete the Plugwise Notification for %s",
77
- coordinator.api.smile_name,
78
- )
79
-
80
- await hass.config_entries.async_forward_entry_setups(entry, PLATFORMS_GATEWAY)
81
-
82
- for component in PLATFORMS_GATEWAY: # pw-beta
83
- if component == Platform.BINARY_SENSOR:
84
- hass.services.async_register(
85
- DOMAIN, SERVICE_DELETE, delete_notification, schema=vol.Schema({})
86
- )
87
 
88
  return True
89
 
90
 
91
- async def _update_listener(
92
- hass: HomeAssistant, entry: ConfigEntry
93
- ) -> None: # pragma: no cover # pw-beta
94
- """Handle options update."""
95
- await hass.config_entries.async_reload(entry.entry_id)
96
-
97
-
98
- async def async_unload_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
99
- """Unload a config entry."""
100
- if unload_ok := await hass.config_entries.async_unload_platforms(
101
- entry, PLATFORMS_GATEWAY
102
- ):
103
- hass.data[DOMAIN].pop(entry.entry_id)
104
- return unload_ok
105
 
106
 
107
  @callback
108
  def async_migrate_entity_entry(entry: er.RegistryEntry) -> dict[str, Any] | None:
109
  """Migrate Plugwise entity entries.
110
 
111
- - Migrates unique ID from old relay switches to the new unique ID
112
  """
113
- if entry.domain == SWITCH_DOMAIN and entry.unique_id.endswith("-"):
 
 
 
 
 
 
 
 
 
 
 
 
 
 
114
  return {"new_unique_id": entry.unique_id.replace("-plug", "-relay")}
115
 
116
- if entry.domain == NUMBER_DOMAIN and entry.unique_id.endswith("-domestic_hot_water_setpoint"):
117
- return {"new_unique_id": entry.unique_id.replace("-domestic_hot_water_setpoint", "-max_dhw_temperature")}
118
-
119
  # No migration needed
120
  return None
121
 
@@ -127,10 +77,10 @@
127
  """Migrate Sensors if needed."""
128
  ent_reg = er.async_get(hass)
129
 
130
- # Migrate opentherm_outdoor_temperature to # pw-beta add to Core
131
- # opentherm_outdoor_air_temperature sensor
132
- for device_id, device in coordinator.data.devices.items():
133
- if device["dev_class"] != "heater_central": # pw-beta add to Core
134
  continue
135
 
136
  old_unique_id = f"{device_id}-outdoor_temperature"
@@ -138,4 +88,10 @@
138
  Platform.SENSOR, DOMAIN, old_unique_id
139
  ):
140
  new_unique_id = f"{device_id}-outdoor_air_temperature"
 
 
 
 
 
 
141
  ent_reg.async_update_entity(entity_id, new_unique_id=new_unique_id)
 
1
  """Plugwise platform for Home Assistant Core."""
2
+
3
  from __future__ import annotations
4
 
5
  from typing import Any
6
 
 
 
 
 
 
7
  from homeassistant.const import Platform
8
+ from homeassistant.core import HomeAssistant, callback
9
  from homeassistant.helpers import device_registry as dr, entity_registry as er
 
10
 
11
+ from .const import DOMAIN, LOGGER, PLATFORMS
12
+ from .coordinator import PlugwiseConfigEntry, PlugwiseDataUpdateCoordinator
 
 
 
 
 
 
 
 
13
 
14
 
15
+ async def async_setup_entry(hass: HomeAssistant, entry: PlugwiseConfigEntry) -> bool:
16
+ """Set up Plugwise components from a config entry."""
17
  await er.async_migrate_entries(hass, entry.entry_id, async_migrate_entity_entry)
18
 
19
+ coordinator = PlugwiseDataUpdateCoordinator(hass, entry)
 
 
 
 
 
 
 
 
 
20
  await coordinator.async_config_entry_first_refresh()
 
21
  migrate_sensor_entities(hass, coordinator)
22
 
23
+ entry.runtime_data = coordinator
 
 
 
 
 
24
 
25
  device_registry = dr.async_get(hass)
26
  device_registry.async_get_or_create(
 
28
  identifiers={(DOMAIN, str(coordinator.api.gateway_id))},
29
  manufacturer="Plugwise",
30
  model=coordinator.api.smile_model,
31
+ model_id=coordinator.api.smile_model_id,
32
  name=coordinator.api.smile_name,
33
+ sw_version=str(coordinator.api.smile_version),
34
+ ) # required for adding the entity-less P1 Gateway
35
 
36
+ await hass.config_entries.async_forward_entry_setups(entry, PLATFORMS)
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
37
 
38
  return True
39
 
40
 
41
+ async def async_unload_entry(hass: HomeAssistant, entry: PlugwiseConfigEntry) -> bool:
42
+ """Unload the Plugwise components."""
43
+ return await hass.config_entries.async_unload_platforms(entry, PLATFORMS)
 
 
 
 
 
 
 
 
 
 
 
44
 
45
 
46
  @callback
47
  def async_migrate_entity_entry(entry: er.RegistryEntry) -> dict[str, Any] | None:
48
  """Migrate Plugwise entity entries.
49
 
50
+ - Migrates old unique ID's from old binary_sensors and switches to the new unique ID's
51
  """
52
+ if entry.domain == Platform.BINARY_SENSOR and entry.unique_id.endswith(
53
+ "-slave_boiler_state"
54
+ ):
55
+ return {
56
+ "new_unique_id": entry.unique_id.replace(
57
+ "-slave_boiler_state", "-secondary_boiler_state"
58
+ )
59
+ }
60
+ if entry.domain == Platform.SENSOR and entry.unique_id.endswith(
61
+ "-relative_humidity"
62
+ ):
63
+ return {
64
+ "new_unique_id": entry.unique_id.replace("-relative_humidity", "-humidity")
65
+ }
66
+ if entry.domain == Platform.SWITCH and entry.unique_id.endswith("-plug"):
67
  return {"new_unique_id": entry.unique_id.replace("-plug", "-relay")}
68
 
 
 
 
69
  # No migration needed
70
  return None
71
 
 
77
  """Migrate Sensors if needed."""
78
  ent_reg = er.async_get(hass)
79
 
80
+ # Migrating opentherm_outdoor_temperature
81
+ # to opentherm_outdoor_air_temperature sensor
82
+ for device_id, device in coordinator.data.items():
83
+ if device["dev_class"] != "heater_central":
84
  continue
85
 
86
  old_unique_id = f"{device_id}-outdoor_temperature"
 
88
  Platform.SENSOR, DOMAIN, old_unique_id
89
  ):
90
  new_unique_id = f"{device_id}-outdoor_air_temperature"
91
+ LOGGER.debug(
92
+ "Migrating entity %s from old unique ID '%s' to new unique ID '%s'",
93
+ entity_id,
94
+ old_unique_id,
95
+ new_unique_id,
96
+ )
97
  ent_reg.async_update_entity(entity_id, new_unique_id=new_unique_id)
/home/runner/work/progress/progress/clones/beta/{beta/custom_components → ha-core/homeassistant/components}/plugwise/binary_sensor.py RENAMED
@@ -1,54 +1,111 @@
1
  """Plugwise Binary Sensor component for Home Assistant."""
 
2
  from __future__ import annotations
3
 
4
  from collections.abc import Mapping
 
5
  from typing import Any
6
 
7
- from homeassistant.components.binary_sensor import BinarySensorEntity
8
- from homeassistant.config_entries import ConfigEntry
9
- from homeassistant.core import HomeAssistant
10
- from homeassistant.helpers.entity_platform import AddEntitiesCallback
11
-
12
- from .const import (
13
- COORDINATOR, # pw-beta
14
- DOMAIN,
15
- LOGGER,
16
- SEVERITIES,
17
  )
18
- from .coordinator import PlugwiseDataUpdateCoordinator
 
 
 
 
19
  from .entity import PlugwiseEntity
20
- from .models import PW_BINARY_SENSOR_TYPES, PlugwiseBinarySensorEntityDescription
21
 
 
 
 
22
  PARALLEL_UPDATES = 0
23
 
24
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
25
  async def async_setup_entry(
26
  hass: HomeAssistant,
27
- config_entry: ConfigEntry,
28
- async_add_entities: AddEntitiesCallback,
29
  ) -> None:
30
- """Set up Plugwise binary sensor based on config_entry."""
31
- coordinator: PlugwiseDataUpdateCoordinator = hass.data[DOMAIN][
32
- config_entry.entry_id
33
- ][COORDINATOR]
34
-
35
- entities: list[PlugwiseBinarySensorEntity] = []
36
- for device_id, device in coordinator.data.devices.items():
37
- if not (binary_sensors := device.get("binary_sensors")):
38
- continue
39
- for description in PW_BINARY_SENSOR_TYPES:
40
- if description.key not in binary_sensors:
41
- continue
42
- entities.append(
43
- PlugwiseBinarySensorEntity(
44
- coordinator,
45
- device_id,
46
- description,
47
- )
48
- )
49
- LOGGER.debug("Add %s binary sensor", description.key)
50
 
51
- async_add_entities(entities)
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
52
 
53
 
54
  class PlugwiseBinarySensorEntity(PlugwiseEntity, BinarySensorEntity):
@@ -66,27 +123,11 @@
66
  super().__init__(coordinator, device_id)
67
  self.entity_description = description
68
  self._attr_unique_id = f"{device_id}-{description.key}"
69
- self._notification: dict[str, str] = {} # pw-beta
70
 
71
  @property
72
  def is_on(self) -> bool:
73
  """Return true if the binary sensor is on."""
74
- # pw-beta: show Plugwise notifications as HA persistent notifications
75
- if self._notification:
76
- for notify_id, message in self._notification.items():
77
- self.hass.components.persistent_notification.async_create(
78
- message, "Plugwise Notification:", f"{DOMAIN}.{notify_id}"
79
- )
80
-
81
- # return self.device["binary_sensors"][self.entity_description.key] # type: ignore [literal-required]
82
- return self.entity_description.value_fn(self.device["binary_sensors"])
83
-
84
- @property
85
- def icon(self) -> str | None:
86
- """Return the icon to use in the frontend, if any."""
87
- if (icon_off := self.entity_description.icon_off) and self.is_on is False:
88
- return icon_off
89
- return self.entity_description.icon
90
 
91
  @property
92
  def extra_state_attributes(self) -> Mapping[str, Any] | None:
@@ -94,25 +135,14 @@
94
  if self.entity_description.key != "plugwise_notification":
95
  return None
96
 
97
- # pw-beta adjustment with attrs is to only represent severities *with* content
98
- # not all severities including those without content as empty lists
99
- attrs: dict[str, list[str]] = {} # pw-beta Re-evaluate against Core
100
- self._notification = {} # pw-beta
101
- if notify := self.coordinator.data.gateway["notifications"]:
102
- for notify_id, details in notify.items(): # pw-beta uses notify_id
103
  for msg_type, msg in details.items():
104
  msg_type = msg_type.lower()
105
  if msg_type not in SEVERITIES:
106
- msg_type = "other" # pragma: no cover
107
-
108
- if (
109
- f"{msg_type}_msg" not in attrs
110
- ): # pw-beta Re-evaluate against Core
111
- attrs[f"{msg_type}_msg"] = []
112
  attrs[f"{msg_type}_msg"].append(msg)
113
 
114
- self._notification[
115
- notify_id
116
- ] = f"{msg_type.title()}: {msg}" # pw-beta
117
-
118
  return attrs
 
1
  """Plugwise Binary Sensor component for Home Assistant."""
2
+
3
  from __future__ import annotations
4
 
5
  from collections.abc import Mapping
6
+ from dataclasses import dataclass
7
  from typing import Any
8
 
9
+ from plugwise.constants import BinarySensorType
10
+
11
+ from homeassistant.components.binary_sensor import (
12
+ BinarySensorDeviceClass,
13
+ BinarySensorEntity,
14
+ BinarySensorEntityDescription,
 
 
 
 
15
  )
16
+ from homeassistant.const import EntityCategory
17
+ from homeassistant.core import HomeAssistant, callback
18
+ from homeassistant.helpers.entity_platform import AddConfigEntryEntitiesCallback
19
+
20
+ from .coordinator import PlugwiseConfigEntry, PlugwiseDataUpdateCoordinator
21
  from .entity import PlugwiseEntity
 
22
 
23
+ SEVERITIES = ["other", "info", "warning", "error"]
24
+
25
+ # Coordinator is used to centralize the data updates
26
  PARALLEL_UPDATES = 0
27
 
28
 
29
+ @dataclass(frozen=True)
30
+ class PlugwiseBinarySensorEntityDescription(BinarySensorEntityDescription):
31
+ """Describes a Plugwise binary sensor entity."""
32
+
33
+ key: BinarySensorType
34
+
35
+
36
+ BINARY_SENSORS: tuple[PlugwiseBinarySensorEntityDescription, ...] = (
37
+ PlugwiseBinarySensorEntityDescription(
38
+ key="low_battery",
39
+ device_class=BinarySensorDeviceClass.BATTERY,
40
+ entity_category=EntityCategory.DIAGNOSTIC,
41
+ ),
42
+ PlugwiseBinarySensorEntityDescription(
43
+ key="compressor_state",
44
+ translation_key="compressor_state",
45
+ entity_category=EntityCategory.DIAGNOSTIC,
46
+ ),
47
+ PlugwiseBinarySensorEntityDescription(
48
+ key="cooling_enabled",
49
+ translation_key="cooling_enabled",
50
+ entity_category=EntityCategory.DIAGNOSTIC,
51
+ ),
52
+ PlugwiseBinarySensorEntityDescription(
53
+ key="dhw_state",
54
+ translation_key="dhw_state",
55
+ entity_category=EntityCategory.DIAGNOSTIC,
56
+ ),
57
+ PlugwiseBinarySensorEntityDescription(
58
+ key="flame_state",
59
+ translation_key="flame_state",
60
+ entity_category=EntityCategory.DIAGNOSTIC,
61
+ ),
62
+ PlugwiseBinarySensorEntityDescription(
63
+ key="heating_state",
64
+ translation_key="heating_state",
65
+ entity_category=EntityCategory.DIAGNOSTIC,
66
+ ),
67
+ PlugwiseBinarySensorEntityDescription(
68
+ key="cooling_state",
69
+ translation_key="cooling_state",
70
+ entity_category=EntityCategory.DIAGNOSTIC,
71
+ ),
72
+ PlugwiseBinarySensorEntityDescription(
73
+ key="secondary_boiler_state",
74
+ translation_key="secondary_boiler_state",
75
+ entity_category=EntityCategory.DIAGNOSTIC,
76
+ ),
77
+ PlugwiseBinarySensorEntityDescription(
78
+ key="plugwise_notification",
79
+ translation_key="plugwise_notification",
80
+ entity_category=EntityCategory.DIAGNOSTIC,
81
+ ),
82
+ )
83
+
84
+
85
  async def async_setup_entry(
86
  hass: HomeAssistant,
87
+ entry: PlugwiseConfigEntry,
88
+ async_add_entities: AddConfigEntryEntitiesCallback,
89
  ) -> None:
90
+ """Set up the Smile binary_sensors from a config entry."""
91
+ coordinator = entry.runtime_data
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
92
 
93
+ @callback
94
+ def _add_entities() -> None:
95
+ """Add Entities."""
96
+ if not coordinator.new_devices:
97
+ return
98
+
99
+ async_add_entities(
100
+ PlugwiseBinarySensorEntity(coordinator, device_id, description)
101
+ for device_id in coordinator.new_devices
102
+ if (binary_sensors := coordinator.data[device_id].get("binary_sensors"))
103
+ for description in BINARY_SENSORS
104
+ if description.key in binary_sensors
105
+ )
106
+
107
+ _add_entities()
108
+ entry.async_on_unload(coordinator.async_add_listener(_add_entities))
109
 
110
 
111
  class PlugwiseBinarySensorEntity(PlugwiseEntity, BinarySensorEntity):
 
123
  super().__init__(coordinator, device_id)
124
  self.entity_description = description
125
  self._attr_unique_id = f"{device_id}-{description.key}"
 
126
 
127
  @property
128
  def is_on(self) -> bool:
129
  """Return true if the binary sensor is on."""
130
+ return self.device["binary_sensors"][self.entity_description.key]
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
131
 
132
  @property
133
  def extra_state_attributes(self) -> Mapping[str, Any] | None:
 
135
  if self.entity_description.key != "plugwise_notification":
136
  return None
137
 
138
+ attrs: dict[str, list[str]] = {f"{severity}_msg": [] for severity in SEVERITIES}
139
+ gateway_id = self.coordinator.api.gateway_id
140
+ if notify := self.coordinator.data[gateway_id]["notifications"]:
141
+ for details in notify.values():
 
 
142
  for msg_type, msg in details.items():
143
  msg_type = msg_type.lower()
144
  if msg_type not in SEVERITIES:
145
+ msg_type = "other"
 
 
 
 
 
146
  attrs[f"{msg_type}_msg"].append(msg)
147
 
 
 
 
 
148
  return attrs
/home/runner/work/progress/progress/clones/beta/{beta/custom_components → ha-core/homeassistant/components}/plugwise/climate.py RENAMED
@@ -1,4 +1,5 @@
1
  """Plugwise Climate component for Home Assistant."""
 
2
  from __future__ import annotations
3
 
4
  from typing import Any
@@ -12,95 +13,113 @@
12
  HVACAction,
13
  HVACMode,
14
  )
15
- from homeassistant.components.climate.const import (
16
- PRESET_AWAY, # pw-beta homekit emulation
17
- PRESET_HOME, # pw-beta homekit emulation
18
- )
19
- from homeassistant.config_entries import ConfigEntry
20
  from homeassistant.const import ATTR_TEMPERATURE, UnitOfTemperature
21
- from homeassistant.core import HomeAssistant
22
- from homeassistant.exceptions import HomeAssistantError
23
- from homeassistant.helpers.entity_platform import AddEntitiesCallback
24
-
25
- from .const import (
26
- CONF_HOMEKIT_EMULATION, # pw-beta homekit emulation
27
- COORDINATOR, # pw-beta
28
- DOMAIN,
29
- MASTER_THERMOSTATS,
30
- )
31
- from .coordinator import PlugwiseDataUpdateCoordinator
32
  from .entity import PlugwiseEntity
33
  from .util import plugwise_command
34
 
 
 
35
 
36
  async def async_setup_entry(
37
  hass: HomeAssistant,
38
- config_entry: ConfigEntry,
39
- async_add_entities: AddEntitiesCallback,
40
  ) -> None:
41
  """Set up the Smile Thermostats from a config entry."""
42
- coordinator: PlugwiseDataUpdateCoordinator = hass.data[DOMAIN][
43
- config_entry.entry_id
44
- ][COORDINATOR]
45
-
46
- homekit_enabled: bool = config_entry.options.get(
47
- CONF_HOMEKIT_EMULATION, False
48
- ) # pw-beta homekit emulation
49
-
50
- async_add_entities(
51
- PlugwiseClimateEntity(
52
- coordinator, device_id, homekit_enabled
53
- ) # pw-beta homekit emulation
54
- for device_id, device in coordinator.data.devices.items()
55
- if device["dev_class"] in MASTER_THERMOSTATS
56
- )
 
 
 
 
 
 
 
 
57
 
58
 
59
  class PlugwiseClimateEntity(PlugwiseEntity, ClimateEntity):
60
- """Representation of an Plugwise thermostat."""
61
 
62
- _attr_has_entity_name = True
63
  _attr_temperature_unit = UnitOfTemperature.CELSIUS
64
  _attr_translation_key = DOMAIN
65
 
 
 
66
  def __init__(
67
  self,
68
  coordinator: PlugwiseDataUpdateCoordinator,
69
  device_id: str,
70
- homekit_enabled: bool, # pw-beta homekit emulation
71
  ) -> None:
72
  """Set up the Plugwise API."""
73
  super().__init__(coordinator, device_id)
74
- self._homekit_enabled = homekit_enabled # pw-beta homekit emulation
75
- self._homekit_mode: str | None = None # pw-beta homekit emulation
76
  self._attr_unique_id = f"{device_id}-climate"
77
 
 
 
 
 
 
 
 
78
  # Determine supported features
79
  self._attr_supported_features = ClimateEntityFeature.TARGET_TEMPERATURE
80
- if self.coordinator.data.gateway["cooling_present"]:
 
 
 
81
  self._attr_supported_features = (
82
  ClimateEntityFeature.TARGET_TEMPERATURE_RANGE
83
  )
 
 
 
 
84
  if presets := self.device.get("preset_modes"):
85
  self._attr_supported_features |= ClimateEntityFeature.PRESET_MODE
86
  self._attr_preset_modes = presets
87
 
88
- # Determine hvac modes and current hvac mode
89
- self._attr_hvac_modes = [HVACMode.HEAT]
90
- if self.coordinator.data.gateway["cooling_present"]:
91
- self._attr_hvac_modes = [HVACMode.HEAT_COOL]
92
- if self.device["available_schedules"] != ["None"]:
93
- self._attr_hvac_modes.append(HVACMode.AUTO)
94
- if self._homekit_enabled: # pw-beta homekit emulation
95
- self._attr_hvac_modes.append(HVACMode.OFF) # pragma: no cover
96
-
97
  self._attr_min_temp = self.device["thermostat"]["lower_bound"]
98
- self._attr_max_temp = self.device["thermostat"]["upper_bound"]
99
- # Fix unpractical resolution provided by Plugwise
100
  self._attr_target_temperature_step = max(
101
- self.device["thermostat"]["resolution"], 0.5
102
  )
103
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
104
  @property
105
  def current_temperature(self) -> float:
106
  """Return the current temperature."""
@@ -133,53 +152,54 @@
133
 
134
  @property
135
  def hvac_mode(self) -> HVACMode:
136
- """Return HVAC operation ie. auto, heat, heat_cool, or off mode."""
137
  if (
138
- mode := self.device["mode"]
139
- ) is None or mode not in self.hvac_modes: # pw-beta add to Core
140
- return HVACMode.HEAT # pragma: no cover
141
- # pw-beta homekit emulation
142
- if self._homekit_enabled and self._homekit_mode == HVACMode.OFF:
143
- mode = HVACMode.OFF # pragma: no cover
144
-
145
  return HVACMode(mode)
146
 
147
  @property
148
- def hvac_action(self) -> HVACAction: # pw-beta add to Core
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
149
  """Return the current running hvac operation if supported."""
150
- # When control_state is present, prefer this data
151
- if (control_state := self.device.get("control_state")) == "cooling":
152
- return HVACAction.COOLING
153
- # Support preheating state as heating,
154
- # until preheating is added as a separate state
155
- if control_state in ["heating", "preheating"]:
156
- return HVACAction.HEATING
157
- if control_state == "off":
158
- return HVACAction.IDLE
159
-
160
- heater: str | None = self.coordinator.data.gateway["heater_id"]
161
- if heater is not None:
162
- heater_data = self.coordinator.data.devices[heater]
163
- if heater_data["binary_sensors"]["heating_state"]:
164
- return HVACAction.HEATING
165
- if heater_data["binary_sensors"].get("cooling_state", False):
166
- return HVACAction.COOLING
167
 
168
  return HVACAction.IDLE
169
 
170
  @property
171
  def preset_mode(self) -> str | None:
172
  """Return the current preset mode."""
173
- return self.device["active_preset"]
174
 
175
  @plugwise_command
176
  async def async_set_temperature(self, **kwargs: Any) -> None:
177
  """Set new target temperature."""
178
- if ATTR_HVAC_MODE in kwargs: # pw-beta add to Core
179
- await self.async_set_hvac_mode(
180
- kwargs[ATTR_HVAC_MODE]
181
- ) # pw-beta add to Core
182
-
183
  data: dict[str, Any] = {}
184
  if ATTR_TEMPERATURE in kwargs:
185
  data["setpoint"] = kwargs.get(ATTR_TEMPERATURE)
@@ -188,38 +208,39 @@
188
  if ATTR_TARGET_TEMP_LOW in kwargs:
189
  data["setpoint_low"] = kwargs.get(ATTR_TARGET_TEMP_LOW)
190
 
191
- for temperature in data.values():
192
- if temperature is None or not (
193
- self._attr_min_temp <= temperature <= self._attr_max_temp
194
- ):
195
- raise ValueError("Invalid temperature change requested")
196
 
197
- await self.coordinator.api.set_temperature(self.device["location"], data)
198
 
199
  @plugwise_command
200
  async def async_set_hvac_mode(self, hvac_mode: HVACMode) -> None:
201
  """Set the hvac mode."""
202
  if hvac_mode not in self.hvac_modes:
203
- raise HomeAssistantError("Unsupported hvac_mode")
 
 
 
 
 
 
 
 
204
 
205
- await self.coordinator.api.set_schedule_state(
206
- self.device["location"],
207
- self.device["last_used"],
208
- "on" if hvac_mode == HVACMode.AUTO else "off",
209
- )
210
 
211
- # pw-beta: feature request - mimic HomeKit behavior
212
- self._homekit_mode = hvac_mode
213
- if self._homekit_enabled:
214
- if self._homekit_mode == HVACMode.OFF: # pragma: no cover
215
- await self.async_set_preset_mode(PRESET_AWAY) # pragma: no cover
216
- if (
217
- self._homekit_mode in [HVACMode.HEAT, HVACMode.COOL, HVACMode.HEAT_COOL]
218
- and self.device["active_preset"] == PRESET_AWAY
219
- ): # pragma: no cover
220
- await self.async_set_preset_mode(PRESET_HOME) # pragma: no cover
221
 
222
  @plugwise_command
223
  async def async_set_preset_mode(self, preset_mode: str) -> None:
224
  """Set the preset mode."""
225
- await self.coordinator.api.set_preset(self.device["location"], preset_mode)
 
1
  """Plugwise Climate component for Home Assistant."""
2
+
3
  from __future__ import annotations
4
 
5
  from typing import Any
 
13
  HVACAction,
14
  HVACMode,
15
  )
 
 
 
 
 
16
  from homeassistant.const import ATTR_TEMPERATURE, UnitOfTemperature
17
+ from homeassistant.core import HomeAssistant, callback
18
+ from homeassistant.exceptions import ServiceValidationError
19
+ from homeassistant.helpers.entity_platform import AddConfigEntryEntitiesCallback
20
+
21
+ from .const import DOMAIN, MASTER_THERMOSTATS
22
+ from .coordinator import PlugwiseConfigEntry, PlugwiseDataUpdateCoordinator
 
 
 
 
 
23
  from .entity import PlugwiseEntity
24
  from .util import plugwise_command
25
 
26
+ PARALLEL_UPDATES = 0
27
+
28
 
29
  async def async_setup_entry(
30
  hass: HomeAssistant,
31
+ entry: PlugwiseConfigEntry,
32
+ async_add_entities: AddConfigEntryEntitiesCallback,
33
  ) -> None:
34
  """Set up the Smile Thermostats from a config entry."""
35
+ coordinator = entry.runtime_data
36
+
37
+ @callback
38
+ def _add_entities() -> None:
39
+ """Add Entities."""
40
+ if not coordinator.new_devices:
41
+ return
42
+
43
+ if coordinator.api.smile_name == "Adam":
44
+ async_add_entities(
45
+ PlugwiseClimateEntity(coordinator, device_id)
46
+ for device_id in coordinator.new_devices
47
+ if coordinator.data[device_id]["dev_class"] == "climate"
48
+ )
49
+ else:
50
+ async_add_entities(
51
+ PlugwiseClimateEntity(coordinator, device_id)
52
+ for device_id in coordinator.new_devices
53
+ if coordinator.data[device_id]["dev_class"] in MASTER_THERMOSTATS
54
+ )
55
+
56
+ _add_entities()
57
+ entry.async_on_unload(coordinator.async_add_listener(_add_entities))
58
 
59
 
60
  class PlugwiseClimateEntity(PlugwiseEntity, ClimateEntity):
61
+ """Representation of a Plugwise thermostat."""
62
 
63
+ _attr_name = None
64
  _attr_temperature_unit = UnitOfTemperature.CELSIUS
65
  _attr_translation_key = DOMAIN
66
 
67
+ _previous_mode: str = "heating"
68
+
69
  def __init__(
70
  self,
71
  coordinator: PlugwiseDataUpdateCoordinator,
72
  device_id: str,
 
73
  ) -> None:
74
  """Set up the Plugwise API."""
75
  super().__init__(coordinator, device_id)
 
 
76
  self._attr_unique_id = f"{device_id}-climate"
77
 
78
+ gateway_id: str = coordinator.api.gateway_id
79
+ self._gateway_data = coordinator.data[gateway_id]
80
+
81
+ self._location = device_id
82
+ if (location := self.device.get("location")) is not None:
83
+ self._location = location
84
+
85
  # Determine supported features
86
  self._attr_supported_features = ClimateEntityFeature.TARGET_TEMPERATURE
87
+ if (
88
+ self.coordinator.api.cooling_present
89
+ and coordinator.api.smile_name != "Adam"
90
+ ):
91
  self._attr_supported_features = (
92
  ClimateEntityFeature.TARGET_TEMPERATURE_RANGE
93
  )
94
+ if HVACMode.OFF in self.hvac_modes:
95
+ self._attr_supported_features |= (
96
+ ClimateEntityFeature.TURN_OFF | ClimateEntityFeature.TURN_ON
97
+ )
98
  if presets := self.device.get("preset_modes"):
99
  self._attr_supported_features |= ClimateEntityFeature.PRESET_MODE
100
  self._attr_preset_modes = presets
101
 
 
 
 
 
 
 
 
 
 
102
  self._attr_min_temp = self.device["thermostat"]["lower_bound"]
103
+ self._attr_max_temp = min(self.device["thermostat"]["upper_bound"], 35.0)
104
+ # Ensure we don't drop below 0.1
105
  self._attr_target_temperature_step = max(
106
+ self.device["thermostat"]["resolution"], 0.1
107
  )
108
 
109
+ def _previous_action_mode(self, coordinator: PlugwiseDataUpdateCoordinator) -> None:
110
+ """Return the previous action-mode when the regulation-mode is not heating or cooling.
111
+
112
+ Helper for set_hvac_mode().
113
+ """
114
+ # When no cooling available, _previous_mode is always heating
115
+ if (
116
+ "regulation_modes" in self._gateway_data
117
+ and "cooling" in self._gateway_data["regulation_modes"]
118
+ ):
119
+ mode = self._gateway_data["select_regulation_mode"]
120
+ if mode in ("cooling", "heating"):
121
+ self._previous_mode = mode
122
+
123
  @property
124
  def current_temperature(self) -> float:
125
  """Return the current temperature."""
 
152
 
153
  @property
154
  def hvac_mode(self) -> HVACMode:
155
+ """Return HVAC operation ie. auto, cool, heat, heat_cool, or off mode."""
156
  if (
157
+ mode := self.device.get("climate_mode")
158
+ ) is None or mode not in self.hvac_modes:
159
+ return HVACMode.HEAT
 
 
 
 
160
  return HVACMode(mode)
161
 
162
  @property
163
+ def hvac_modes(self) -> list[HVACMode]:
164
+ """Return a list of available HVACModes."""
165
+ hvac_modes: list[HVACMode] = []
166
+ if "regulation_modes" in self._gateway_data:
167
+ hvac_modes.append(HVACMode.OFF)
168
+
169
+ if "available_schedules" in self.device:
170
+ hvac_modes.append(HVACMode.AUTO)
171
+
172
+ if self.coordinator.api.cooling_present:
173
+ if "regulation_modes" in self._gateway_data:
174
+ if self._gateway_data["select_regulation_mode"] == "cooling":
175
+ hvac_modes.append(HVACMode.COOL)
176
+ if self._gateway_data["select_regulation_mode"] == "heating":
177
+ hvac_modes.append(HVACMode.HEAT)
178
+ else:
179
+ hvac_modes.append(HVACMode.HEAT_COOL)
180
+ else:
181
+ hvac_modes.append(HVACMode.HEAT)
182
+
183
+ return hvac_modes
184
+
185
+ @property
186
+ def hvac_action(self) -> HVACAction:
187
  """Return the current running hvac operation if supported."""
188
+ # Keep track of the previous action-mode
189
+ self._previous_action_mode(self.coordinator)
190
+ if (action := self.device.get("control_state")) is not None:
191
+ return HVACAction(action)
 
 
 
 
 
 
 
 
 
 
 
 
 
192
 
193
  return HVACAction.IDLE
194
 
195
  @property
196
  def preset_mode(self) -> str | None:
197
  """Return the current preset mode."""
198
+ return self.device.get("active_preset")
199
 
200
  @plugwise_command
201
  async def async_set_temperature(self, **kwargs: Any) -> None:
202
  """Set new target temperature."""
 
 
 
 
 
203
  data: dict[str, Any] = {}
204
  if ATTR_TEMPERATURE in kwargs:
205
  data["setpoint"] = kwargs.get(ATTR_TEMPERATURE)
 
208
  if ATTR_TARGET_TEMP_LOW in kwargs:
209
  data["setpoint_low"] = kwargs.get(ATTR_TARGET_TEMP_LOW)
210
 
211
+ if mode := kwargs.get(ATTR_HVAC_MODE):
212
+ await self.async_set_hvac_mode(mode)
 
 
 
213
 
214
+ await self.coordinator.api.set_temperature(self._location, data)
215
 
216
  @plugwise_command
217
  async def async_set_hvac_mode(self, hvac_mode: HVACMode) -> None:
218
  """Set the hvac mode."""
219
  if hvac_mode not in self.hvac_modes:
220
+ hvac_modes = ", ".join(self.hvac_modes)
221
+ raise ServiceValidationError(
222
+ translation_domain=DOMAIN,
223
+ translation_key="unsupported_hvac_mode_requested",
224
+ translation_placeholders={
225
+ "hvac_mode": hvac_mode,
226
+ "hvac_modes": hvac_modes,
227
+ },
228
+ )
229
 
230
+ if hvac_mode == self.hvac_mode:
231
+ return
 
 
 
232
 
233
+ if hvac_mode == HVACMode.OFF:
234
+ await self.coordinator.api.set_regulation_mode(hvac_mode)
235
+ else:
236
+ await self.coordinator.api.set_schedule_state(
237
+ self._location,
238
+ "on" if hvac_mode == HVACMode.AUTO else "off",
239
+ )
240
+ if self.hvac_mode == HVACMode.OFF:
241
+ await self.coordinator.api.set_regulation_mode(self._previous_mode)
 
242
 
243
  @plugwise_command
244
  async def async_set_preset_mode(self, preset_mode: str) -> None:
245
  """Set the preset mode."""
246
+ await self.coordinator.api.set_preset(self._location, preset_mode)
/home/runner/work/progress/progress/clones/beta/{beta/custom_components → ha-core/homeassistant/components}/plugwise/config_flow.py RENAMED
@@ -1,43 +1,37 @@
1
  """Config flow for Plugwise integration."""
 
2
  from __future__ import annotations
3
 
4
- import datetime as dt # pw-beta options
5
- from typing import Any
6
 
 
 
 
 
 
 
 
 
 
7
  import voluptuous as vol
8
 
9
- from homeassistant import config_entries
10
- from homeassistant.components.zeroconf import ZeroconfServiceInfo
11
- from homeassistant.config_entries import ConfigEntry, ConfigFlow
12
  from homeassistant.const import (
 
13
  CONF_BASE,
14
  CONF_HOST,
15
  CONF_NAME,
16
  CONF_PASSWORD,
17
  CONF_PORT,
18
- CONF_SCAN_INTERVAL,
19
  CONF_USERNAME,
20
  )
21
- from homeassistant.core import HomeAssistant, callback
22
- from homeassistant.data_entry_flow import FlowResult
23
- from homeassistant.helpers import config_validation as cv
24
  from homeassistant.helpers.aiohttp_client import async_get_clientsession
25
- from plugwise import Smile
26
- from plugwise.exceptions import (
27
- ConnectionFailedError,
28
- InvalidAuthentication,
29
- InvalidSetupError,
30
- InvalidXMLError,
31
- ResponseError,
32
- UnsupportedDeviceError,
33
- )
34
 
35
  from .const import (
36
- CONF_HOMEKIT_EMULATION, # pw-beta option
37
- CONF_REFRESH_INTERVAL, # pw-beta option
38
- COORDINATOR,
39
  DEFAULT_PORT,
40
- DEFAULT_SCAN_INTERVAL, # pw-beta option
41
  DEFAULT_USERNAME,
42
  DOMAIN,
43
  FLOW_SMILE,
@@ -48,42 +42,36 @@
48
  ZEROCONF_MAP,
49
  )
50
 
 
51
 
52
- def _base_gw_schema(
53
- discovery_info: ZeroconfServiceInfo | None,
54
- user_input: dict[str, Any] | None,
55
- ) -> vol.Schema:
 
 
 
 
56
  """Generate base schema for gateways."""
 
 
57
  if not discovery_info:
58
- if not user_input:
59
- return vol.Schema(
60
- {
61
- vol.Required(CONF_PASSWORD): str,
62
- vol.Required(CONF_HOST): str,
63
- vol.Optional(CONF_PORT, default=DEFAULT_PORT): int,
64
- vol.Required(CONF_USERNAME, default=SMILE): vol.In(
65
- {SMILE: FLOW_SMILE, STRETCH: FLOW_STRETCH}
66
- ),
67
- }
68
- )
69
- return vol.Schema(
70
  {
71
- vol.Required(CONF_PASSWORD, default=user_input[CONF_PASSWORD]): str,
72
- vol.Required(CONF_HOST, default=user_input[CONF_HOST]): str,
73
- vol.Optional(CONF_PORT, default=user_input[CONF_PORT]): int,
74
- vol.Required(CONF_USERNAME, default=user_input[CONF_USERNAME]): vol.In(
75
  {SMILE: FLOW_SMILE, STRETCH: FLOW_STRETCH}
76
  ),
77
  }
78
  )
79
 
80
- return vol.Schema({vol.Required(CONF_PASSWORD): str})
81
 
82
 
83
- async def validate_gw_input(hass: HomeAssistant, data: dict[str, Any]) -> Smile:
84
  """Validate whether the user input allows us to connect to the gateway.
85
 
86
- Data has the keys from _base_gw_schema() with values provided by the user.
87
  """
88
  websession = async_get_clientsession(hass, verify_ssl=False)
89
  api = Smile(
@@ -91,24 +79,50 @@
91
  password=data[CONF_PASSWORD],
92
  port=data[CONF_PORT],
93
  username=data[CONF_USERNAME],
94
- timeout=30,
95
  websession=websession,
96
  )
97
  await api.connect()
98
  return api
99
 
100
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
101
  class PlugwiseConfigFlow(ConfigFlow, domain=DOMAIN):
102
  """Handle a config flow for Plugwise Smile."""
103
 
104
  VERSION = 1
105
 
106
  discovery_info: ZeroconfServiceInfo | None = None
 
107
  _username: str = DEFAULT_USERNAME
108
 
109
  async def async_step_zeroconf(
110
  self, discovery_info: ZeroconfServiceInfo
111
- ) -> FlowResult:
112
  """Prepare configuration for a discovered Plugwise Smile."""
113
  self.discovery_info = discovery_info
114
  _properties = discovery_info.properties
@@ -116,7 +130,7 @@
116
  unique_id = discovery_info.hostname.split(".")[0].split("-")[0]
117
  if config_entry := await self.async_set_unique_id(unique_id):
118
  try:
119
- await validate_gw_input(
120
  self.hass,
121
  {
122
  CONF_HOST: discovery_info.host,
@@ -125,7 +139,7 @@
125
  CONF_PASSWORD: config_entry.data[CONF_PASSWORD],
126
  },
127
  )
128
- except Exception: # pylint: disable=broad-except
129
  self._abort_if_unique_id_configured()
130
  else:
131
  self._abort_if_unique_id_configured(
@@ -137,7 +151,7 @@
137
 
138
  if DEFAULT_USERNAME not in unique_id:
139
  self._username = STRETCH_USERNAME
140
- _product = _properties.get("product", None)
141
  _version = _properties.get("version", "n/a")
142
  _name = f"{ZEROCONF_MAP.get(_product, _product)} v{_version}"
143
 
@@ -149,153 +163,94 @@
149
  # If we have discovered an Adam or Anna, both might be on the network.
150
  # In that case, we need to cancel the Anna flow, as the Adam should
151
  # be added.
152
- for flow in self._async_in_progress():
153
- # This is an Anna, and there is already an Adam flow in progress
154
- if (
155
- _product == "smile_thermo"
156
- and "context" in flow
157
- and flow["context"].get("product") == "smile_open_therm"
158
- ):
159
- return self.async_abort(reason="anna_with_adam")
160
-
161
- # This is an Adam, and there is already an Anna flow in progress
162
- if (
163
- _product == "smile_open_therm"
164
- and "context" in flow
165
- and flow["context"].get("product") == "smile_thermo"
166
- and "flow_id" in flow
167
- ):
168
- self.hass.config_entries.flow.async_abort(flow["flow_id"])
169
 
170
  self.context.update(
171
  {
172
- "title_placeholders": {
173
- CONF_HOST: discovery_info.host,
174
- CONF_NAME: _name,
175
- CONF_PORT: discovery_info.port,
176
- CONF_USERNAME: self._username,
177
- },
178
- "configuration_url": f"http://{discovery_info.host}:{discovery_info.port}",
179
- "product": _product,
180
  }
181
  )
182
  return await self.async_step_user()
183
 
 
 
 
 
 
 
 
 
 
 
 
 
184
  async def async_step_user(
185
  self, user_input: dict[str, Any] | None = None
186
- ) -> FlowResult:
187
  """Handle the initial step when using network/gateway setups."""
188
  errors: dict[str, str] = {}
189
 
190
- if not user_input:
191
- return self.async_show_form(
192
- step_id="user",
193
- data_schema=_base_gw_schema(self.discovery_info, None),
194
- errors=errors,
195
- )
196
-
197
- if self.discovery_info:
198
- user_input[CONF_HOST] = self.discovery_info.host
199
- user_input[CONF_PORT] = self.discovery_info.port
200
- user_input[CONF_USERNAME] = self._username
201
- try:
202
- api = await validate_gw_input(self.hass, user_input)
203
- except ConnectionFailedError:
204
- errors[CONF_BASE] = "cannot_connect"
205
- except InvalidAuthentication:
206
- errors[CONF_BASE] = "invalid_auth"
207
- except InvalidSetupError:
208
- errors[CONF_BASE] = "invalid_setup"
209
- except (InvalidXMLError, ResponseError):
210
- errors[CONF_BASE] = "response_error"
211
- except UnsupportedDeviceError:
212
- errors[CONF_BASE] = "unsupported"
213
- except Exception: # pylint: disable=broad-except
214
- errors[CONF_BASE] = "unknown"
215
-
216
- if errors:
217
- return self.async_show_form(
218
- step_id="user",
219
- data_schema=_base_gw_schema(None, user_input),
220
- errors=errors,
221
- )
222
-
223
- await self.async_set_unique_id(
224
- api.smile_hostname or api.gateway_id, raise_on_progress=False
225
- )
226
- self._abort_if_unique_id_configured()
227
-
228
- return self.async_create_entry(title=api.smile_name, data=user_input)
229
-
230
- @staticmethod
231
- @callback
232
- def async_get_options_flow(
233
- config_entry: ConfigEntry,
234
- ) -> config_entries.OptionsFlow: # pw-beta options
235
- """Get the options flow for this handler."""
236
- return PlugwiseOptionsFlowHandler(config_entry)
237
-
238
-
239
- # pw-beta - change the scan-interval via CONFIGURE
240
- # pw-beta - add homekit emulation via CONFIGURE
241
- # pw-beta - change the frontend refresh interval via CONFIGURE
242
- class PlugwiseOptionsFlowHandler(config_entries.OptionsFlow): # pw-beta options
243
- """Plugwise option flow."""
244
-
245
- def __init__(self, config_entry: ConfigEntry) -> None: # pragma: no cover
246
- """Initialize options flow."""
247
- self.config_entry = config_entry
248
-
249
- async def async_step_none(
250
- self, user_input: dict[str, Any] | None = None
251
- ) -> FlowResult: # pragma: no cover
252
- """No options available."""
253
  if user_input is not None:
254
- # Apparently not possible to abort an options flow at the moment
255
- return self.async_create_entry(title="", data=self.config_entry.options)
 
 
 
 
 
 
 
 
 
 
 
 
256
 
257
- return self.async_show_form(step_id="none")
 
 
 
 
258
 
259
- async def async_step_init(
260
  self, user_input: dict[str, Any] | None = None
261
- ) -> FlowResult: # pragma: no cover
262
- """Manage the Plugwise options."""
263
- if not self.config_entry.data.get(CONF_HOST):
264
- return await self.async_step_none(user_input)
265
-
266
- if user_input is not None:
267
- return self.async_create_entry(title="", data=user_input)
268
-
269
- coordinator = self.hass.data[DOMAIN][self.config_entry.entry_id][COORDINATOR]
270
- interval: dt.timedelta = DEFAULT_SCAN_INTERVAL[
271
- coordinator.api.smile_type
272
- ] # pw-beta options
273
-
274
- data = {
275
- vol.Optional(
276
- CONF_SCAN_INTERVAL,
277
- default=self.config_entry.options.get(
278
- CONF_SCAN_INTERVAL, interval.seconds
279
- ),
280
- ): vol.All(cv.positive_int, vol.Clamp(min=10)),
281
- } # pw-beta
282
 
283
- if coordinator.api.smile_type != "thermostat":
284
- return self.async_show_form(step_id="init", data_schema=vol.Schema(data))
285
 
286
- data.update(
287
- {
288
- vol.Optional(
289
- CONF_HOMEKIT_EMULATION,
290
- default=self.config_entry.options.get(
291
- CONF_HOMEKIT_EMULATION, False
292
- ),
293
- ): cv.boolean,
294
- vol.Optional(
295
- CONF_REFRESH_INTERVAL,
296
- default=self.config_entry.options.get(CONF_REFRESH_INTERVAL, 1.5),
297
- ): vol.All(vol.Coerce(float), vol.Range(min=1.5, max=10.0)),
298
  }
299
- ) # pw-beta
300
 
301
- return self.async_show_form(step_id="init", data_schema=vol.Schema(data))
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
  """Config flow for Plugwise integration."""
2
+
3
  from __future__ import annotations
4
 
5
+ import logging
6
+ from typing import Any, Self
7
 
8
+ from plugwise import Smile
9
+ from plugwise.exceptions import (
10
+ ConnectionFailedError,
11
+ InvalidAuthentication,
12
+ InvalidSetupError,
13
+ InvalidXMLError,
14
+ ResponseError,
15
+ UnsupportedDeviceError,
16
+ )
17
  import voluptuous as vol
18
 
19
+ from homeassistant.config_entries import SOURCE_USER, ConfigFlow, ConfigFlowResult
 
 
20
  from homeassistant.const import (
21
+ ATTR_CONFIGURATION_URL,
22
  CONF_BASE,
23
  CONF_HOST,
24
  CONF_NAME,
25
  CONF_PASSWORD,
26
  CONF_PORT,
 
27
  CONF_USERNAME,
28
  )
29
+ from homeassistant.core import HomeAssistant
 
 
30
  from homeassistant.helpers.aiohttp_client import async_get_clientsession
31
+ from homeassistant.helpers.service_info.zeroconf import ZeroconfServiceInfo
 
 
 
 
 
 
 
 
32
 
33
  from .const import (
 
 
 
34
  DEFAULT_PORT,
 
35
  DEFAULT_USERNAME,
36
  DOMAIN,
37
  FLOW_SMILE,
 
42
  ZEROCONF_MAP,
43
  )
44
 
45
+ _LOGGER = logging.getLogger(__name__)
46
 
47
+ SMILE_RECONF_SCHEMA = vol.Schema(
48
+ {
49
+ vol.Required(CONF_HOST): str,
50
+ }
51
+ )
52
+
53
+
54
+ def smile_user_schema(discovery_info: ZeroconfServiceInfo | None) -> vol.Schema:
55
  """Generate base schema for gateways."""
56
+ schema = vol.Schema({vol.Required(CONF_PASSWORD): str})
57
+
58
  if not discovery_info:
59
+ schema = schema.extend(
 
 
 
 
 
 
 
 
 
 
 
60
  {
61
+ vol.Required(CONF_HOST): str,
62
+ vol.Required(CONF_USERNAME, default=SMILE): vol.In(
 
 
63
  {SMILE: FLOW_SMILE, STRETCH: FLOW_STRETCH}
64
  ),
65
  }
66
  )
67
 
68
+ return schema
69
 
70
 
71
+ async def validate_input(hass: HomeAssistant, data: dict[str, Any]) -> Smile:
72
  """Validate whether the user input allows us to connect to the gateway.
73
 
74
+ Data has the keys from the schema with values provided by the user.
75
  """
76
  websession = async_get_clientsession(hass, verify_ssl=False)
77
  api = Smile(
 
79
  password=data[CONF_PASSWORD],
80
  port=data[CONF_PORT],
81
  username=data[CONF_USERNAME],
 
82
  websession=websession,
83
  )
84
  await api.connect()
85
  return api
86
 
87
 
88
+ async def verify_connection(
89
+ hass: HomeAssistant, user_input: dict[str, Any]
90
+ ) -> tuple[Smile | None, dict[str, str]]:
91
+ """Verify and return the gateway connection or an error."""
92
+ errors: dict[str, str] = {}
93
+
94
+ try:
95
+ return (await validate_input(hass, user_input), errors)
96
+ except ConnectionFailedError:
97
+ errors[CONF_BASE] = "cannot_connect"
98
+ except InvalidAuthentication:
99
+ errors[CONF_BASE] = "invalid_auth"
100
+ except InvalidSetupError:
101
+ errors[CONF_BASE] = "invalid_setup"
102
+ except (InvalidXMLError, ResponseError):
103
+ errors[CONF_BASE] = "response_error"
104
+ except UnsupportedDeviceError:
105
+ errors[CONF_BASE] = "unsupported"
106
+ except Exception:
107
+ _LOGGER.exception(
108
+ "Unknown exception while verifying connection with your Plugwise Smile"
109
+ )
110
+ errors[CONF_BASE] = "unknown"
111
+ return (None, errors)
112
+
113
+
114
  class PlugwiseConfigFlow(ConfigFlow, domain=DOMAIN):
115
  """Handle a config flow for Plugwise Smile."""
116
 
117
  VERSION = 1
118
 
119
  discovery_info: ZeroconfServiceInfo | None = None
120
+ product: str = "Unknown Smile"
121
  _username: str = DEFAULT_USERNAME
122
 
123
  async def async_step_zeroconf(
124
  self, discovery_info: ZeroconfServiceInfo
125
+ ) -> ConfigFlowResult:
126
  """Prepare configuration for a discovered Plugwise Smile."""
127
  self.discovery_info = discovery_info
128
  _properties = discovery_info.properties
 
130
  unique_id = discovery_info.hostname.split(".")[0].split("-")[0]
131
  if config_entry := await self.async_set_unique_id(unique_id):
132
  try:
133
+ await validate_input(
134
  self.hass,
135
  {
136
  CONF_HOST: discovery_info.host,
 
139
  CONF_PASSWORD: config_entry.data[CONF_PASSWORD],
140
  },
141
  )
142
+ except Exception: # noqa: BLE001
143
  self._abort_if_unique_id_configured()
144
  else:
145
  self._abort_if_unique_id_configured(
 
151
 
152
  if DEFAULT_USERNAME not in unique_id:
153
  self._username = STRETCH_USERNAME
154
+ self.product = _product = _properties.get("product", "Unknown Smile")
155
  _version = _properties.get("version", "n/a")
156
  _name = f"{ZEROCONF_MAP.get(_product, _product)} v{_version}"
157
 
 
163
  # If we have discovered an Adam or Anna, both might be on the network.
164
  # In that case, we need to cancel the Anna flow, as the Adam should
165
  # be added.
166
+ if self.hass.config_entries.flow.async_has_matching_flow(self):
167
+ return self.async_abort(reason="anna_with_adam")
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
168
 
169
  self.context.update(
170
  {
171
+ "title_placeholders": {CONF_NAME: _name},
172
+ ATTR_CONFIGURATION_URL: (
173
+ f"http://{discovery_info.host}:{discovery_info.port}"
174
+ ),
 
 
 
 
175
  }
176
  )
177
  return await self.async_step_user()
178
 
179
+ def is_matching(self, other_flow: Self) -> bool:
180
+ """Return True if other_flow is matching this flow."""
181
+ # This is an Anna, and there is already an Adam flow in progress
182
+ if self.product == "smile_thermo" and other_flow.product == "smile_open_therm":
183
+ return True
184
+
185
+ # This is an Adam, and there is already an Anna flow in progress
186
+ if self.product == "smile_open_therm" and other_flow.product == "smile_thermo":
187
+ self.hass.config_entries.flow.async_abort(other_flow.flow_id)
188
+
189
+ return False
190
+
191
  async def async_step_user(
192
  self, user_input: dict[str, Any] | None = None
193
+ ) -> ConfigFlowResult:
194
  """Handle the initial step when using network/gateway setups."""
195
  errors: dict[str, str] = {}
196
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
197
  if user_input is not None:
198
+ user_input[CONF_PORT] = DEFAULT_PORT
199
+ if self.discovery_info:
200
+ user_input[CONF_HOST] = self.discovery_info.host
201
+ user_input[CONF_PORT] = self.discovery_info.port
202
+ user_input[CONF_USERNAME] = self._username
203
+
204
+ api, errors = await verify_connection(self.hass, user_input)
205
+ if api:
206
+ await self.async_set_unique_id(
207
+ api.smile_hostname or api.gateway_id,
208
+ raise_on_progress=False,
209
+ )
210
+ self._abort_if_unique_id_configured()
211
+ return self.async_create_entry(title=api.smile_name, data=user_input)
212
 
213
+ return self.async_show_form(
214
+ step_id=SOURCE_USER,
215
+ data_schema=smile_user_schema(self.discovery_info),
216
+ errors=errors,
217
+ )
218
 
219
+ async def async_step_reconfigure(
220
  self, user_input: dict[str, Any] | None = None
221
+ ) -> ConfigFlowResult:
222
+ """Handle reconfiguration of the integration."""
223
+ errors: dict[str, str] = {}
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
224
 
225
+ reconfigure_entry = self._get_reconfigure_entry()
 
226
 
227
+ if user_input:
228
+ # Keep current username and password
229
+ full_input = {
230
+ CONF_HOST: user_input.get(CONF_HOST),
231
+ CONF_PORT: reconfigure_entry.data.get(CONF_PORT),
232
+ CONF_USERNAME: reconfigure_entry.data.get(CONF_USERNAME),
233
+ CONF_PASSWORD: reconfigure_entry.data.get(CONF_PASSWORD),
 
 
 
 
 
234
  }
 
235
 
236
+ api, errors = await verify_connection(self.hass, full_input)
237
+ if api:
238
+ await self.async_set_unique_id(
239
+ api.smile_hostname or api.gateway_id,
240
+ raise_on_progress=False,
241
+ )
242
+ self._abort_if_unique_id_mismatch(reason="not_the_same_smile")
243
+ return self.async_update_reload_and_abort(
244
+ reconfigure_entry,
245
+ data_updates=full_input,
246
+ )
247
+
248
+ return self.async_show_form(
249
+ step_id="reconfigure",
250
+ data_schema=self.add_suggested_values_to_schema(
251
+ data_schema=SMILE_RECONF_SCHEMA,
252
+ suggested_values=reconfigure_entry.data,
253
+ ),
254
+ description_placeholders={"title": reconfigure_entry.title},
255
+ errors=errors,
256
+ )
/home/runner/work/progress/progress/clones/beta/{beta/custom_components → ha-core/homeassistant/components}/plugwise/const.py RENAMED
@@ -1,7 +1,10 @@
1
  """Constants for Plugwise component."""
 
 
 
2
  from datetime import timedelta
3
  import logging
4
- from typing import Final
5
 
6
  from homeassistant.const import Platform
7
 
@@ -10,56 +13,66 @@
10
  LOGGER = logging.getLogger(__package__)
11
 
12
  API: Final = "api"
13
- COORDINATOR: Final = "coordinator"
14
- CONF_HOMEKIT_EMULATION: Final = "homekit_emulation" # pw-beta options
15
- CONF_REFRESH_INTERVAL: Final = "refresh_interval" # pw-beta options
16
- CONF_MANUAL_PATH: Final = "Enter Manually"
 
 
 
17
  SMILE: Final = "smile"
18
  STRETCH: Final = "stretch"
19
  STRETCH_USERNAME: Final = "stretch"
20
 
21
- FLOW_NET: Final = "Network: Smile/Stretch"
22
- FLOW_SMILE: Final = "Smile (Adam/Anna/P1)"
23
- FLOW_STRETCH: Final = "Stretch (Stretch)"
24
- FLOW_TYPE: Final = "flow_type"
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
25
 
26
- UNDO_UPDATE_LISTENER: Final = "undo_update_listener"
 
 
 
 
 
 
 
 
 
 
 
27
 
28
  # Default directives
 
 
29
  DEFAULT_PORT: Final = 80
30
  DEFAULT_SCAN_INTERVAL: Final[dict[str, timedelta]] = {
31
  "power": timedelta(seconds=10),
32
  "stretch": timedelta(seconds=60),
33
  "thermostat": timedelta(seconds=60),
34
  }
35
- DEFAULT_TIMEOUT: Final = 10
36
  DEFAULT_USERNAME: Final = "smile"
37
 
38
- # --- Const for Plugwise Smile and Stretch
39
- PLATFORMS_GATEWAY: Final[list[str]] = [
40
- Platform.BINARY_SENSOR,
41
- Platform.CLIMATE,
42
- Platform.NUMBER,
43
- Platform.SELECT,
44
- Platform.SENSOR,
45
- Platform.SWITCH,
46
- ]
47
- SENSOR_PLATFORMS: Final[list[str]] = [Platform.SENSOR, Platform.SWITCH]
48
- SERVICE_DELETE: Final = "delete_notification"
49
- SEVERITIES: Final[list[str]] = ["other", "info", "message", "warning", "error"]
50
-
51
- # Climate const:
52
  MASTER_THERMOSTATS: Final[list[str]] = [
53
  "thermostat",
 
54
  "zone_thermometer",
55
  "zone_thermostat",
56
- "thermostatic_radiator_valve",
57
  ]
58
-
59
- # Config_flow const:
60
- ZEROCONF_MAP: Final[dict[str, str]] = {
61
- "smile": "Smile P1",
62
- "smile_thermo": "Smile Anna",
63
- "smile_open_therm": "Adam",
64
- "stretch": "Stretch",
65
- }
 
1
  """Constants for Plugwise component."""
2
+
3
+ from __future__ import annotations
4
+
5
  from datetime import timedelta
6
  import logging
7
+ from typing import Final, Literal
8
 
9
  from homeassistant.const import Platform
10
 
 
13
  LOGGER = logging.getLogger(__package__)
14
 
15
  API: Final = "api"
16
+ FLOW_SMILE: Final = "smile (Adam/Anna/P1)"
17
+ FLOW_STRETCH: Final = "stretch (Stretch)"
18
+ FLOW_TYPE: Final = "flow_type"
19
+ GATEWAY: Final = "gateway"
20
+ LOCATION: Final = "location"
21
+ PW_TYPE: Final = "plugwise_type"
22
+ REBOOT: Final = "reboot"
23
  SMILE: Final = "smile"
24
  STRETCH: Final = "stretch"
25
  STRETCH_USERNAME: Final = "stretch"
26
 
27
+ PLATFORMS: Final[list[str]] = [
28
+ Platform.BINARY_SENSOR,
29
+ Platform.BUTTON,
30
+ Platform.CLIMATE,
31
+ Platform.NUMBER,
32
+ Platform.SELECT,
33
+ Platform.SENSOR,
34
+ Platform.SWITCH,
35
+ ]
36
+ ZEROCONF_MAP: Final[dict[str, str]] = {
37
+ "smile": "Smile P1",
38
+ "smile_thermo": "Smile Anna",
39
+ "smile_open_therm": "Adam",
40
+ "stretch": "Stretch",
41
+ }
42
+
43
+ type NumberType = Literal[
44
+ "maximum_boiler_temperature",
45
+ "max_dhw_temperature",
46
+ "temperature_offset",
47
+ ]
48
 
49
+ type SelectType = Literal[
50
+ "select_dhw_mode",
51
+ "select_gateway_mode",
52
+ "select_regulation_mode",
53
+ "select_schedule",
54
+ ]
55
+ type SelectOptionsType = Literal[
56
+ "dhw_modes",
57
+ "gateway_modes",
58
+ "regulation_modes",
59
+ "available_schedules",
60
+ ]
61
 
62
  # Default directives
63
+ DEFAULT_MAX_TEMP: Final = 30
64
+ DEFAULT_MIN_TEMP: Final = 4
65
  DEFAULT_PORT: Final = 80
66
  DEFAULT_SCAN_INTERVAL: Final[dict[str, timedelta]] = {
67
  "power": timedelta(seconds=10),
68
  "stretch": timedelta(seconds=60),
69
  "thermostat": timedelta(seconds=60),
70
  }
 
71
  DEFAULT_USERNAME: Final = "smile"
72
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
73
  MASTER_THERMOSTATS: Final[list[str]] = [
74
  "thermostat",
75
+ "thermostatic_radiator_valve",
76
  "zone_thermometer",
77
  "zone_thermostat",
 
78
  ]
 
 
 
 
 
 
 
 
/home/runner/work/progress/progress/clones/beta/{beta/custom_components → ha-core/homeassistant/components}/plugwise/coordinator.py RENAMED
@@ -1,115 +1,146 @@
1
  """DataUpdateCoordinator for Plugwise."""
 
2
  from datetime import timedelta
3
 
4
- from homeassistant.config_entries import ConfigEntry
5
- from homeassistant.const import (
6
- CONF_HOST,
7
- CONF_PASSWORD,
8
- CONF_PORT,
9
- CONF_SCAN_INTERVAL, # pw-beta options
10
- CONF_USERNAME,
11
- )
12
- from homeassistant.core import HomeAssistant
13
- from homeassistant.exceptions import ConfigEntryError
14
- from homeassistant.helpers.aiohttp_client import async_get_clientsession
15
- from homeassistant.helpers.debounce import Debouncer
16
- from homeassistant.helpers.update_coordinator import DataUpdateCoordinator, UpdateFailed
17
- from plugwise import PlugwiseData, Smile
18
  from plugwise.exceptions import (
19
  ConnectionFailedError,
20
  InvalidAuthentication,
21
  InvalidXMLError,
 
22
  ResponseError,
23
  UnsupportedDeviceError,
24
  )
25
 
26
- from .const import DEFAULT_PORT, DEFAULT_SCAN_INTERVAL, DEFAULT_USERNAME, DOMAIN, LOGGER
 
 
 
 
 
 
 
 
 
27
 
 
28
 
29
- class PlugwiseDataUpdateCoordinator(DataUpdateCoordinator[PlugwiseData]):
 
30
  """Class to manage fetching Plugwise data from single endpoint."""
31
 
32
  _connected: bool = False
33
 
34
- def __init__(
35
- self,
36
- hass: HomeAssistant,
37
- entry: ConfigEntry,
38
- cooldown: float,
39
- update_interval: timedelta = timedelta(seconds=60),
40
- ) -> None: # pw-beta cooldown
41
  """Initialize the coordinator."""
42
  super().__init__(
43
  hass,
44
  LOGGER,
 
45
  name=DOMAIN,
46
- # Core directly updates from const's DEFAULT_SCAN_INTERVAL
47
- update_interval=update_interval,
48
  # Don't refresh immediately, give the device time to process
49
  # the change in state before we query it.
50
  request_refresh_debouncer=Debouncer(
51
  hass,
52
  LOGGER,
53
- cooldown=cooldown,
54
  immediate=False,
55
  ),
56
  )
57
 
58
  self.api = Smile(
59
- host=entry.data[CONF_HOST],
60
- username=entry.data.get(CONF_USERNAME, DEFAULT_USERNAME),
61
- password=entry.data[CONF_PASSWORD],
62
- port=entry.data.get(CONF_PORT, DEFAULT_PORT),
63
- timeout=30,
64
  websession=async_get_clientsession(hass, verify_ssl=False),
65
  )
66
- self._entry = entry
67
- self._unavailable_logged = False
68
- self.update_interval = update_interval
69
 
70
  async def _connect(self) -> None:
71
  """Connect to the Plugwise Smile."""
72
- self._connected = await self.api.connect()
73
- self.api.get_all_devices()
74
-
75
- self.update_interval = DEFAULT_SCAN_INTERVAL.get(
76
- self.api.smile_type, timedelta(seconds=60)
77
- ) # pw-beta options scan-interval
78
- if (custom_time := self._entry.options.get(CONF_SCAN_INTERVAL)) is not None:
79
- self.update_interval = timedelta(
80
- seconds=int(custom_time)
81
- ) # pragma: no cover # pw-beta options
82
 
83
- LOGGER.debug("DUC update interval: %s", self.update_interval) # pw-beta options
84
-
85
- async def _async_update_data(self) -> PlugwiseData:
86
  """Fetch data from Plugwise."""
87
- data = PlugwiseData(gateway={}, devices={})
88
-
89
  try:
90
  if not self._connected:
91
  await self._connect()
92
  data = await self.api.async_update()
93
- LOGGER.debug(f"{self.api.smile_name} data: %s", data)
94
- if self._unavailable_logged:
95
- self._unavailable_logged = False
 
 
96
  except InvalidAuthentication as err:
97
- if not self._unavailable_logged: # pw-beta add to Core
98
- self._unavailable_logged = True
99
- raise ConfigEntryError("Authentication failed") from err
 
100
  except (InvalidXMLError, ResponseError) as err:
101
- if not self._unavailable_logged: # pw-beta add to Core
102
- self._unavailable_logged = True
103
- raise UpdateFailed(
104
- "Invalid XML data, or error indication received from the Plugwise Adam/Smile/Stretch"
105
- ) from err
 
 
 
 
106
  except UnsupportedDeviceError as err:
107
- if not self._unavailable_logged: # pw-beta add to Core
108
- self._unavailable_logged = True
109
- raise ConfigEntryError("Device with unsupported firmware") from err
110
- except ConnectionFailedError as err:
111
- if not self._unavailable_logged: # pw-beta add to Core
112
- self._unavailable_logged = True
113
- raise UpdateFailed("Failed to connect") from err
114
 
 
115
  return data
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
  """DataUpdateCoordinator for Plugwise."""
2
+
3
  from datetime import timedelta
4
 
5
+ from packaging.version import Version
6
+ from plugwise import GwEntityData, Smile
 
 
 
 
 
 
 
 
 
 
 
 
7
  from plugwise.exceptions import (
8
  ConnectionFailedError,
9
  InvalidAuthentication,
10
  InvalidXMLError,
11
+ PlugwiseError,
12
  ResponseError,
13
  UnsupportedDeviceError,
14
  )
15
 
16
+ from homeassistant.config_entries import ConfigEntry
17
+ from homeassistant.const import CONF_HOST, CONF_PASSWORD, CONF_PORT, CONF_USERNAME
18
+ from homeassistant.core import HomeAssistant
19
+ from homeassistant.exceptions import ConfigEntryError
20
+ from homeassistant.helpers import device_registry as dr
21
+ from homeassistant.helpers.aiohttp_client import async_get_clientsession
22
+ from homeassistant.helpers.debounce import Debouncer
23
+ from homeassistant.helpers.update_coordinator import DataUpdateCoordinator, UpdateFailed
24
+
25
+ from .const import DEFAULT_PORT, DEFAULT_USERNAME, DOMAIN, LOGGER
26
 
27
+ type PlugwiseConfigEntry = ConfigEntry[PlugwiseDataUpdateCoordinator]
28
 
29
+
30
+ class PlugwiseDataUpdateCoordinator(DataUpdateCoordinator[dict[str, GwEntityData]]):
31
  """Class to manage fetching Plugwise data from single endpoint."""
32
 
33
  _connected: bool = False
34
 
35
+ config_entry: PlugwiseConfigEntry
36
+
37
+ def __init__(self, hass: HomeAssistant, config_entry: PlugwiseConfigEntry) -> None:
 
 
 
 
38
  """Initialize the coordinator."""
39
  super().__init__(
40
  hass,
41
  LOGGER,
42
+ config_entry=config_entry,
43
  name=DOMAIN,
44
+ update_interval=timedelta(seconds=60),
 
45
  # Don't refresh immediately, give the device time to process
46
  # the change in state before we query it.
47
  request_refresh_debouncer=Debouncer(
48
  hass,
49
  LOGGER,
50
+ cooldown=1.5,
51
  immediate=False,
52
  ),
53
  )
54
 
55
  self.api = Smile(
56
+ host=self.config_entry.data[CONF_HOST],
57
+ username=self.config_entry.data.get(CONF_USERNAME, DEFAULT_USERNAME),
58
+ password=self.config_entry.data[CONF_PASSWORD],
59
+ port=self.config_entry.data.get(CONF_PORT, DEFAULT_PORT),
 
60
  websession=async_get_clientsession(hass, verify_ssl=False),
61
  )
62
+ self._current_devices: set[str] = set()
63
+ self.new_devices: set[str] = set()
 
64
 
65
  async def _connect(self) -> None:
66
  """Connect to the Plugwise Smile."""
67
+ version = await self.api.connect()
68
+ self._connected = isinstance(version, Version)
 
 
 
 
 
 
 
 
69
 
70
+ async def _async_update_data(self) -> dict[str, GwEntityData]:
 
 
71
  """Fetch data from Plugwise."""
 
 
72
  try:
73
  if not self._connected:
74
  await self._connect()
75
  data = await self.api.async_update()
76
+ except ConnectionFailedError as err:
77
+ raise UpdateFailed(
78
+ translation_domain=DOMAIN,
79
+ translation_key="failed_to_connect",
80
+ ) from err
81
  except InvalidAuthentication as err:
82
+ raise ConfigEntryError(
83
+ translation_domain=DOMAIN,
84
+ translation_key="authentication_failed",
85
+ ) from err
86
  except (InvalidXMLError, ResponseError) as err:
87
+ raise UpdateFailed(
88
+ translation_domain=DOMAIN,
89
+ translation_key="invalid_xml_data",
90
+ ) from err
91
+ except PlugwiseError as err:
92
+ raise UpdateFailed(
93
+ translation_domain=DOMAIN,
94
+ translation_key="data_incomplete_or_missing",
95
+ ) from err
96
  except UnsupportedDeviceError as err:
97
+ raise ConfigEntryError(
98
+ translation_domain=DOMAIN,
99
+ translation_key="unsupported_firmware",
100
+ ) from err
 
 
 
101
 
102
+ self._async_add_remove_devices(data, self.config_entry)
103
  return data
104
+
105
+ def _async_add_remove_devices(
106
+ self, data: dict[str, GwEntityData], entry: ConfigEntry
107
+ ) -> None:
108
+ """Add new Plugwise devices, remove non-existing devices."""
109
+ # Check for new or removed devices
110
+ self.new_devices = set(data) - self._current_devices
111
+ removed_devices = self._current_devices - set(data)
112
+ self._current_devices = set(data)
113
+
114
+ if removed_devices:
115
+ self._async_remove_devices(data, entry)
116
+
117
+ def _async_remove_devices(
118
+ self, data: dict[str, GwEntityData], entry: ConfigEntry
119
+ ) -> None:
120
+ """Clean registries when removed devices found."""
121
+ device_reg = dr.async_get(self.hass)
122
+ device_list = dr.async_entries_for_config_entry(
123
+ device_reg, self.config_entry.entry_id
124
+ )
125
+ # First find the Plugwise via_device
126
+ gateway_device = device_reg.async_get_device({(DOMAIN, self.api.gateway_id)})
127
+ assert gateway_device is not None
128
+ via_device_id = gateway_device.id
129
+
130
+ # Then remove the connected orphaned device(s)
131
+ for device_entry in device_list:
132
+ for identifier in device_entry.identifiers:
133
+ if identifier[0] == DOMAIN:
134
+ if (
135
+ device_entry.via_device_id == via_device_id
136
+ and identifier[1] not in data
137
+ ):
138
+ device_reg.async_update_device(
139
+ device_entry.id, remove_config_entry_id=entry.entry_id
140
+ )
141
+ LOGGER.debug(
142
+ "Removed %s device %s %s from device_registry",
143
+ DOMAIN,
144
+ device_entry.model,
145
+ identifier[1],
146
+ )
/home/runner/work/progress/progress/clones/beta/{beta/custom_components → ha-core/homeassistant/components}/plugwise/diagnostics.py RENAMED
@@ -1,26 +1,17 @@
1
  """Diagnostics support for Plugwise."""
 
2
  from __future__ import annotations
3
 
4
  from typing import Any
5
 
6
- from homeassistant.config_entries import ConfigEntry
7
  from homeassistant.core import HomeAssistant
8
 
9
- from .const import (
10
- COORDINATOR, # pw-beta
11
- DOMAIN,
12
- )
13
- from .coordinator import PlugwiseDataUpdateCoordinator
14
 
15
 
16
  async def async_get_config_entry_diagnostics(
17
- hass: HomeAssistant, entry: ConfigEntry
18
  ) -> dict[str, Any]:
19
  """Return diagnostics for a config entry."""
20
- coordinator: PlugwiseDataUpdateCoordinator = hass.data[DOMAIN][entry.entry_id][
21
- COORDINATOR
22
- ]
23
- return {
24
- "gateway": coordinator.data.gateway,
25
- "devices": coordinator.data.devices,
26
- }
 
1
  """Diagnostics support for Plugwise."""
2
+
3
  from __future__ import annotations
4
 
5
  from typing import Any
6
 
 
7
  from homeassistant.core import HomeAssistant
8
 
9
+ from .coordinator import PlugwiseConfigEntry
 
 
 
 
10
 
11
 
12
  async def async_get_config_entry_diagnostics(
13
+ hass: HomeAssistant, entry: PlugwiseConfigEntry
14
  ) -> dict[str, Any]:
15
  """Return diagnostics for a config entry."""
16
+ coordinator = entry.runtime_data
17
+ return coordinator.data
 
 
 
 
 
/home/runner/work/progress/progress/clones/beta/{beta/custom_components → ha-core/homeassistant/components}/plugwise/entity.py RENAMED
@@ -1,14 +1,16 @@
1
  """Generic Plugwise Entity Class."""
 
2
  from __future__ import annotations
3
 
 
 
4
  from homeassistant.const import ATTR_NAME, ATTR_VIA_DEVICE, CONF_HOST
5
  from homeassistant.helpers.device_registry import (
6
  CONNECTION_NETWORK_MAC,
7
  CONNECTION_ZIGBEE,
 
8
  )
9
- from homeassistant.helpers.entity import DeviceInfo
10
  from homeassistant.helpers.update_coordinator import CoordinatorEntity
11
- from plugwise.constants import DeviceData
12
 
13
  from .const import DOMAIN
14
  from .coordinator import PlugwiseDataUpdateCoordinator
@@ -32,7 +34,7 @@
32
  if entry := self.coordinator.config_entry:
33
  configuration_url = f"http://{entry.data[CONF_HOST]}"
34
 
35
- data = coordinator.data.devices[device_id]
36
  connections = set()
37
  if mac := data.get("mac_address"):
38
  connections.add((CONNECTION_NETWORK_MAC, mac))
@@ -45,18 +47,19 @@
45
  connections=connections,
46
  manufacturer=data.get("vendor"),
47
  model=data.get("model"),
48
- name=coordinator.data.gateway["smile_name"],
 
49
  sw_version=data.get("firmware"),
50
  hw_version=data.get("hardware"),
51
  )
52
 
53
- if device_id != coordinator.data.gateway["gateway_id"]:
54
  self._attr_device_info.update(
55
  {
56
  ATTR_NAME: data.get("name"),
57
  ATTR_VIA_DEVICE: (
58
  DOMAIN,
59
- str(self.coordinator.data.gateway["gateway_id"]),
60
  ),
61
  }
62
  )
@@ -65,17 +68,12 @@
65
  def available(self) -> bool:
66
  """Return if entity is available."""
67
  return (
68
- self._dev_id in self.coordinator.data.devices
69
  and ("available" not in self.device or self.device["available"] is True)
70
  and super().available
71
  )
72
 
73
  @property
74
- def device(self) -> DeviceData:
75
  """Return data for this device."""
76
- return self.coordinator.data.devices[self._dev_id]
77
-
78
- async def async_added_to_hass(self) -> None:
79
- """Subscribe to updates."""
80
- self._handle_coordinator_update()
81
- await super().async_added_to_hass()
 
1
  """Generic Plugwise Entity Class."""
2
+
3
  from __future__ import annotations
4
 
5
+ from plugwise.constants import GwEntityData
6
+
7
  from homeassistant.const import ATTR_NAME, ATTR_VIA_DEVICE, CONF_HOST
8
  from homeassistant.helpers.device_registry import (
9
  CONNECTION_NETWORK_MAC,
10
  CONNECTION_ZIGBEE,
11
+ DeviceInfo,
12
  )
 
13
  from homeassistant.helpers.update_coordinator import CoordinatorEntity
 
14
 
15
  from .const import DOMAIN
16
  from .coordinator import PlugwiseDataUpdateCoordinator
 
34
  if entry := self.coordinator.config_entry:
35
  configuration_url = f"http://{entry.data[CONF_HOST]}"
36
 
37
+ data = coordinator.data[device_id]
38
  connections = set()
39
  if mac := data.get("mac_address"):
40
  connections.add((CONNECTION_NETWORK_MAC, mac))
 
47
  connections=connections,
48
  manufacturer=data.get("vendor"),
49
  model=data.get("model"),
50
+ model_id=data.get("model_id"),
51
+ name=coordinator.api.smile_name,
52
  sw_version=data.get("firmware"),
53
  hw_version=data.get("hardware"),
54
  )
55
 
56
+ if device_id != coordinator.api.gateway_id:
57
  self._attr_device_info.update(
58
  {
59
  ATTR_NAME: data.get("name"),
60
  ATTR_VIA_DEVICE: (
61
  DOMAIN,
62
+ str(self.coordinator.api.gateway_id),
63
  ),
64
  }
65
  )
 
68
  def available(self) -> bool:
69
  """Return if entity is available."""
70
  return (
71
+ self._dev_id in self.coordinator.data
72
  and ("available" not in self.device or self.device["available"] is True)
73
  and super().available
74
  )
75
 
76
  @property
77
+ def device(self) -> GwEntityData:
78
  """Return data for this device."""
79
+ return self.coordinator.data[self._dev_id]
 
 
 
 
 
/home/runner/work/progress/progress/clones/beta/{beta/custom_components → ha-core/homeassistant/components}/plugwise/manifest.json RENAMED
@@ -1,13 +1,13 @@
1
  {
2
  "domain": "plugwise",
3
- "name": "Plugwise Smile/Stretch Beta",
4
- "after_dependencies": ["zeroconf"],
5
  "codeowners": ["@CoMPaTech", "@bouwew"],
6
  "config_flow": true,
7
- "documentation": "https://github.com/plugwise/plugwise-beta",
8
  "integration_type": "hub",
9
  "iot_class": "local_polling",
10
  "loggers": ["plugwise"],
11
- "requirements": ["plugwise==0.31.6"],
12
- "version": "0.40.3a6"
 
13
  }
 
1
  {
2
  "domain": "plugwise",
3
+ "name": "Plugwise",
 
4
  "codeowners": ["@CoMPaTech", "@bouwew"],
5
  "config_flow": true,
6
+ "documentation": "https://www.home-assistant.io/integrations/plugwise",
7
  "integration_type": "hub",
8
  "iot_class": "local_polling",
9
  "loggers": ["plugwise"],
10
+ "quality_scale": "platinum",
11
+ "requirements": ["plugwise==1.7.3"],
12
+ "zeroconf": ["_plugwise._tcp.local."]
13
  }
/home/runner/work/progress/progress/clones/beta/{beta/custom_components → ha-core/homeassistant/components}/plugwise/number.py RENAMED
@@ -1,7 +1,7 @@
1
  """Number platform for Plugwise integration."""
 
2
  from __future__ import annotations
3
 
4
- from collections.abc import Awaitable, Callable
5
  from dataclasses import dataclass
6
 
7
  from homeassistant.components.number import (
@@ -10,89 +10,73 @@
10
  NumberEntityDescription,
11
  NumberMode,
12
  )
13
- from homeassistant.config_entries import ConfigEntry
14
  from homeassistant.const import EntityCategory, UnitOfTemperature
15
- from homeassistant.core import HomeAssistant
16
- from homeassistant.helpers.entity_platform import AddEntitiesCallback
17
- from plugwise import ActuatorData, DeviceData, Smile
18
-
19
- from .const import (
20
- COORDINATOR, # pw-beta
21
- DOMAIN,
22
- LOGGER,
23
- )
24
- from .coordinator import PlugwiseDataUpdateCoordinator
25
- from .entity import PlugwiseEntity
26
 
 
 
 
 
27
 
28
- @dataclass
29
- class PlugwiseNumberMixin:
30
- """Mixin values for Plugwse entities."""
31
-
32
- command: Callable[[Smile, str, float], Awaitable[None]]
33
- native_max_value_fn: Callable[[ActuatorData], float]
34
- native_min_value_fn: Callable[[ActuatorData], float]
35
- native_step_fn: Callable[[ActuatorData], float]
36
- native_value_fn: Callable[[ActuatorData], float]
37
- actuator_fn: Callable[[DeviceData], ActuatorData | None]
38
 
39
 
40
- @dataclass
41
- class PlugwiseNumberEntityDescription(NumberEntityDescription, PlugwiseNumberMixin):
42
  """Class describing Plugwise Number entities."""
43
 
 
 
44
 
45
  NUMBER_TYPES = (
46
  PlugwiseNumberEntityDescription(
47
  key="maximum_boiler_temperature",
48
  translation_key="maximum_boiler_temperature",
49
- command=lambda api, number, value: api.set_number_setpoint(number, value),
50
  device_class=NumberDeviceClass.TEMPERATURE,
51
  entity_category=EntityCategory.CONFIG,
52
  native_unit_of_measurement=UnitOfTemperature.CELSIUS,
53
- native_max_value_fn=lambda data: data["upper_bound"],
54
- native_min_value_fn=lambda data: data["lower_bound"],
55
- native_step_fn=lambda data: data["resolution"],
56
- native_value_fn=lambda data: data["setpoint"],
57
- actuator_fn=lambda data: data.get("maximum_boiler_temperature"),
58
  ),
59
  PlugwiseNumberEntityDescription(
60
  key="max_dhw_temperature",
61
  translation_key="max_dhw_temperature",
62
- command=lambda api, number, value: api.set_number_setpoint(number, value),
63
  device_class=NumberDeviceClass.TEMPERATURE,
64
  entity_category=EntityCategory.CONFIG,
65
  native_unit_of_measurement=UnitOfTemperature.CELSIUS,
66
- native_max_value_fn=lambda data: data["upper_bound"],
67
- native_min_value_fn=lambda data: data["lower_bound"],
68
- native_step_fn=lambda data: data["resolution"],
69
- native_value_fn=lambda data: data["setpoint"],
70
- actuator_fn=lambda data: data.get("max_dhw_temperature"),
 
 
71
  ),
72
  )
73
 
74
 
75
  async def async_setup_entry(
76
  hass: HomeAssistant,
77
- config_entry: ConfigEntry,
78
- async_add_entities: AddEntitiesCallback,
79
  ) -> None:
80
  """Set up Plugwise number platform."""
 
81
 
82
- coordinator: PlugwiseDataUpdateCoordinator = hass.data[DOMAIN][
83
- config_entry.entry_id
84
- ][COORDINATOR]
85
-
86
- entities: list[PlugwiseNumberEntity] = []
87
- for device_id, device in coordinator.data.devices.items():
88
- for description in NUMBER_TYPES:
89
- if actuator := description.actuator_fn(device):
90
- entities.append(
91
- PlugwiseNumberEntity(actuator, coordinator, device_id, description)
92
- )
93
- LOGGER.debug("Add %s %s number", device["name"], description.name)
94
 
95
- async_add_entities(entities)
 
96
 
97
 
98
  class PlugwiseNumberEntity(PlugwiseEntity, NumberEntity):
@@ -102,44 +86,32 @@
102
 
103
  def __init__(
104
  self,
105
- actuator: ActuatorData,
106
  coordinator: PlugwiseDataUpdateCoordinator,
107
  device_id: str,
108
  description: PlugwiseNumberEntityDescription,
109
  ) -> None:
110
  """Initiate Plugwise Number."""
111
  super().__init__(coordinator, device_id)
112
- self.actuator = actuator
113
- self.entity_description = description
114
- self._attr_unique_id = f"{device_id}-{description.key}"
115
  self._attr_mode = NumberMode.BOX
 
 
 
 
 
116
 
117
- @property
118
- def native_max_value(self) -> float:
119
- """Return the setpoint max. value."""
120
- return self.entity_description.native_max_value_fn(self.actuator)
121
-
122
- @property
123
- def native_min_value(self) -> float:
124
- """Return the setpoint min. value."""
125
- return self.entity_description.native_min_value_fn(self.actuator)
126
-
127
- @property
128
- def native_step(self) -> float:
129
- """Return the setpoint step value."""
130
- return max(self.entity_description.native_step_fn(self.actuator), 0.5)
131
 
132
  @property
133
  def native_value(self) -> float:
134
  """Return the present setpoint value."""
135
- return self.entity_description.native_value_fn(self.actuator)
136
 
 
137
  async def async_set_native_value(self, value: float) -> None:
138
  """Change to the new setpoint value."""
139
- await self.entity_description.command(
140
- self.coordinator.api, self.entity_description.key, value
141
- )
142
- LOGGER.debug(
143
- "Setting %s to %s was successful", self.entity_description.name, value
144
  )
145
- await self.coordinator.async_request_refresh()
 
1
  """Number platform for Plugwise integration."""
2
+
3
  from __future__ import annotations
4
 
 
5
  from dataclasses import dataclass
6
 
7
  from homeassistant.components.number import (
 
10
  NumberEntityDescription,
11
  NumberMode,
12
  )
 
13
  from homeassistant.const import EntityCategory, UnitOfTemperature
14
+ from homeassistant.core import HomeAssistant, callback
15
+ from homeassistant.helpers.entity_platform import AddConfigEntryEntitiesCallback
 
 
 
 
 
 
 
 
 
16
 
17
+ from .const import NumberType
18
+ from .coordinator import PlugwiseConfigEntry, PlugwiseDataUpdateCoordinator
19
+ from .entity import PlugwiseEntity
20
+ from .util import plugwise_command
21
 
22
+ PARALLEL_UPDATES = 0
 
 
 
 
 
 
 
 
 
23
 
24
 
25
+ @dataclass(frozen=True, kw_only=True)
26
+ class PlugwiseNumberEntityDescription(NumberEntityDescription):
27
  """Class describing Plugwise Number entities."""
28
 
29
+ key: NumberType
30
+
31
 
32
  NUMBER_TYPES = (
33
  PlugwiseNumberEntityDescription(
34
  key="maximum_boiler_temperature",
35
  translation_key="maximum_boiler_temperature",
 
36
  device_class=NumberDeviceClass.TEMPERATURE,
37
  entity_category=EntityCategory.CONFIG,
38
  native_unit_of_measurement=UnitOfTemperature.CELSIUS,
 
 
 
 
 
39
  ),
40
  PlugwiseNumberEntityDescription(
41
  key="max_dhw_temperature",
42
  translation_key="max_dhw_temperature",
 
43
  device_class=NumberDeviceClass.TEMPERATURE,
44
  entity_category=EntityCategory.CONFIG,
45
  native_unit_of_measurement=UnitOfTemperature.CELSIUS,
46
+ ),
47
+ PlugwiseNumberEntityDescription(
48
+ key="temperature_offset",
49
+ translation_key="temperature_offset",
50
+ device_class=NumberDeviceClass.TEMPERATURE,
51
+ entity_category=EntityCategory.CONFIG,
52
+ native_unit_of_measurement=UnitOfTemperature.CELSIUS,
53
  ),
54
  )
55
 
56
 
57
  async def async_setup_entry(
58
  hass: HomeAssistant,
59
+ entry: PlugwiseConfigEntry,
60
+ async_add_entities: AddConfigEntryEntitiesCallback,
61
  ) -> None:
62
  """Set up Plugwise number platform."""
63
+ coordinator = entry.runtime_data
64
 
65
+ @callback
66
+ def _add_entities() -> None:
67
+ """Add Entities."""
68
+ if not coordinator.new_devices:
69
+ return
70
+
71
+ async_add_entities(
72
+ PlugwiseNumberEntity(coordinator, device_id, description)
73
+ for device_id in coordinator.new_devices
74
+ for description in NUMBER_TYPES
75
+ if description.key in coordinator.data[device_id]
76
+ )
77
 
78
+ _add_entities()
79
+ entry.async_on_unload(coordinator.async_add_listener(_add_entities))
80
 
81
 
82
  class PlugwiseNumberEntity(PlugwiseEntity, NumberEntity):
 
86
 
87
  def __init__(
88
  self,
 
89
  coordinator: PlugwiseDataUpdateCoordinator,
90
  device_id: str,
91
  description: PlugwiseNumberEntityDescription,
92
  ) -> None:
93
  """Initiate Plugwise Number."""
94
  super().__init__(coordinator, device_id)
 
 
 
95
  self._attr_mode = NumberMode.BOX
96
+ self._attr_native_max_value = self.device[description.key]["upper_bound"]
97
+ self._attr_native_min_value = self.device[description.key]["lower_bound"]
98
+ self._attr_unique_id = f"{device_id}-{description.key}"
99
+ self.device_id = device_id
100
+ self.entity_description = description
101
 
102
+ native_step = self.device[description.key]["resolution"]
103
+ if description.key != "temperature_offset":
104
+ native_step = max(native_step, 0.5)
105
+ self._attr_native_step = native_step
 
 
 
 
 
 
 
 
 
 
106
 
107
  @property
108
  def native_value(self) -> float:
109
  """Return the present setpoint value."""
110
+ return self.device[self.entity_description.key]["setpoint"]
111
 
112
+ @plugwise_command
113
  async def async_set_native_value(self, value: float) -> None:
114
  """Change to the new setpoint value."""
115
+ await self.coordinator.api.set_number(
116
+ self.device_id, self.entity_description.key, value
 
 
 
117
  )
 
/home/runner/work/progress/progress/clones/beta/{beta/custom_components → ha-core/homeassistant/components}/plugwise/select.py RENAMED
@@ -1,93 +1,80 @@
1
  """Plugwise Select component for Home Assistant."""
 
2
  from __future__ import annotations
3
 
4
- from collections.abc import Awaitable, Callable
5
  from dataclasses import dataclass
6
 
7
  from homeassistant.components.select import SelectEntity, SelectEntityDescription
8
- from homeassistant.config_entries import ConfigEntry
9
  from homeassistant.const import STATE_ON, EntityCategory
10
- from homeassistant.core import HomeAssistant
11
- from homeassistant.helpers.entity_platform import AddEntitiesCallback
12
- from plugwise import DeviceData, Smile
13
-
14
- from .const import (
15
- COORDINATOR, # pw-beta
16
- DOMAIN,
17
- LOGGER,
18
- )
19
- from .coordinator import PlugwiseDataUpdateCoordinator
20
  from .entity import PlugwiseEntity
 
21
 
22
  PARALLEL_UPDATES = 0
23
 
24
 
25
- @dataclass
26
- class PlugwiseSelectDescriptionMixin:
27
- """Mixin values for Plugwise Select entities."""
28
-
29
- command: Callable[[Smile, str, str], Awaitable[None]]
30
- value_fn: Callable[[DeviceData], str]
31
- options_fn: Callable[[DeviceData], list[str] | None]
32
-
33
 
34
- @dataclass
35
- class PlugwiseSelectEntityDescription(
36
- SelectEntityDescription, PlugwiseSelectDescriptionMixin
37
- ):
38
- """Class describing Plugwise Number entities."""
39
 
40
 
41
  SELECT_TYPES = (
42
  PlugwiseSelectEntityDescription(
43
  key="select_schedule",
44
- translation_key="thermostat_schedule",
45
- icon="mdi:calendar-clock",
46
- command=lambda api, loc, opt: api.set_schedule_state(loc, opt, STATE_ON),
47
- value_fn=lambda data: data["selected_schedule"],
48
- options_fn=lambda data: data.get("available_schedules"),
49
  ),
50
  PlugwiseSelectEntityDescription(
51
  key="select_regulation_mode",
52
  translation_key="regulation_mode",
53
- icon="mdi:hvac",
54
  entity_category=EntityCategory.CONFIG,
55
- command=lambda api, loc, opt: api.set_regulation_mode(opt),
56
- value_fn=lambda data: data["regulation_mode"],
57
- options_fn=lambda data: data.get("regulation_modes"),
58
  ),
59
  PlugwiseSelectEntityDescription(
60
  key="select_dhw_mode",
61
  translation_key="dhw_mode",
62
- icon="mdi:shower",
63
  entity_category=EntityCategory.CONFIG,
64
- command=lambda api, loc, opt: api.set_dhw_mode(opt),
65
- value_fn=lambda data: data["dhw_mode"],
66
- options_fn=lambda data: data.get("dhw_modes"),
 
 
 
 
67
  ),
68
  )
69
 
70
 
71
  async def async_setup_entry(
72
  hass: HomeAssistant,
73
- config_entry: ConfigEntry,
74
- async_add_entities: AddEntitiesCallback,
75
  ) -> None:
76
  """Set up the Smile selector from a config entry."""
77
- coordinator: PlugwiseDataUpdateCoordinator = hass.data[DOMAIN][
78
- config_entry.entry_id
79
- ][COORDINATOR]
80
-
81
- entities: list[PlugwiseSelectEntity] = []
82
- for device_id, device in coordinator.data.devices.items():
83
- for description in SELECT_TYPES:
84
- if (options := description.options_fn(device)) and len(options) > 1:
85
- entities.append(
86
- PlugwiseSelectEntity(coordinator, device_id, description, options)
87
- )
88
- LOGGER.debug("Add %s %s selector", device["name"], description.name)
 
 
89
 
90
- async_add_entities(entities)
 
91
 
92
 
93
  class PlugwiseSelectEntity(PlugwiseEntity, SelectEntity):
@@ -100,28 +87,32 @@
100
  coordinator: PlugwiseDataUpdateCoordinator,
101
  device_id: str,
102
  entity_description: PlugwiseSelectEntityDescription,
103
- options: list[str],
104
  ) -> None:
105
  """Initialise the selector."""
106
  super().__init__(coordinator, device_id)
107
- self.entity_description = entity_description
108
  self._attr_unique_id = f"{device_id}-{entity_description.key}"
109
- self._attr_options = options
 
 
 
 
110
 
111
  @property
112
  def current_option(self) -> str:
113
  """Return the selected entity option to represent the entity state."""
114
- # return self.device[self.entity_description.current_option_key] # type: ignore [literal-required]
115
- return self.entity_description.value_fn(self.device)
116
 
 
 
 
 
 
 
117
  async def async_select_option(self, option: str) -> None:
118
- """Change to the selected entity option."""
119
- await self.entity_description.command(
120
- self.coordinator.api, self.device["location"], option
121
- )
122
- LOGGER.debug(
123
- "Set %s to %s was successful.",
124
- self.entity_description.name,
125
- option,
126
  )
127
- await self.coordinator.async_request_refresh()
 
1
  """Plugwise Select component for Home Assistant."""
2
+
3
  from __future__ import annotations
4
 
 
5
  from dataclasses import dataclass
6
 
7
  from homeassistant.components.select import SelectEntity, SelectEntityDescription
 
8
  from homeassistant.const import STATE_ON, EntityCategory
9
+ from homeassistant.core import HomeAssistant, callback
10
+ from homeassistant.helpers.entity_platform import AddConfigEntryEntitiesCallback
11
+
12
+ from .const import SelectOptionsType, SelectType
13
+ from .coordinator import PlugwiseConfigEntry, PlugwiseDataUpdateCoordinator
 
 
 
 
 
14
  from .entity import PlugwiseEntity
15
+ from .util import plugwise_command
16
 
17
  PARALLEL_UPDATES = 0
18
 
19
 
20
+ @dataclass(frozen=True, kw_only=True)
21
+ class PlugwiseSelectEntityDescription(SelectEntityDescription):
22
+ """Class describing Plugwise Select entities."""
 
 
 
 
 
23
 
24
+ key: SelectType
25
+ options_key: SelectOptionsType
 
 
 
26
 
27
 
28
  SELECT_TYPES = (
29
  PlugwiseSelectEntityDescription(
30
  key="select_schedule",
31
+ translation_key="select_schedule",
32
+ options_key="available_schedules",
 
 
 
33
  ),
34
  PlugwiseSelectEntityDescription(
35
  key="select_regulation_mode",
36
  translation_key="regulation_mode",
 
37
  entity_category=EntityCategory.CONFIG,
38
+ options_key="regulation_modes",
 
 
39
  ),
40
  PlugwiseSelectEntityDescription(
41
  key="select_dhw_mode",
42
  translation_key="dhw_mode",
 
43
  entity_category=EntityCategory.CONFIG,
44
+ options_key="dhw_modes",
45
+ ),
46
+ PlugwiseSelectEntityDescription(
47
+ key="select_gateway_mode",
48
+ translation_key="gateway_mode",
49
+ entity_category=EntityCategory.CONFIG,
50
+ options_key="gateway_modes",
51
  ),
52
  )
53
 
54
 
55
  async def async_setup_entry(
56
  hass: HomeAssistant,
57
+ entry: PlugwiseConfigEntry,
58
+ async_add_entities: AddConfigEntryEntitiesCallback,
59
  ) -> None:
60
  """Set up the Smile selector from a config entry."""
61
+ coordinator = entry.runtime_data
62
+
63
+ @callback
64
+ def _add_entities() -> None:
65
+ """Add Entities."""
66
+ if not coordinator.new_devices:
67
+ return
68
+
69
+ async_add_entities(
70
+ PlugwiseSelectEntity(coordinator, device_id, description)
71
+ for device_id in coordinator.new_devices
72
+ for description in SELECT_TYPES
73
+ if description.options_key in coordinator.data[device_id]
74
+ )
75
 
76
+ _add_entities()
77
+ entry.async_on_unload(coordinator.async_add_listener(_add_entities))
78
 
79
 
80
  class PlugwiseSelectEntity(PlugwiseEntity, SelectEntity):
 
87
  coordinator: PlugwiseDataUpdateCoordinator,
88
  device_id: str,
89
  entity_description: PlugwiseSelectEntityDescription,
 
90
  ) -> None:
91
  """Initialise the selector."""
92
  super().__init__(coordinator, device_id)
 
93
  self._attr_unique_id = f"{device_id}-{entity_description.key}"
94
+ self.entity_description = entity_description
95
+
96
+ self._location = device_id
97
+ if (location := self.device.get("location")) is not None:
98
+ self._location = location
99
 
100
  @property
101
  def current_option(self) -> str:
102
  """Return the selected entity option to represent the entity state."""
103
+ return self.device[self.entity_description.key]
 
104
 
105
+ @property
106
+ def options(self) -> list[str]:
107
+ """Return the available select-options."""
108
+ return self.device[self.entity_description.options_key]
109
+
110
+ @plugwise_command
111
  async def async_select_option(self, option: str) -> None:
112
+ """Change to the selected entity option.
113
+
114
+ self._location and STATE_ON are required for the thermostat-schedule select.
115
+ """
116
+ await self.coordinator.api.set_select(
117
+ self.entity_description.key, self._location, option, STATE_ON
 
 
118
  )
 
/home/runner/work/progress/progress/clones/beta/{beta/custom_components → ha-core/homeassistant/components}/plugwise/sensor.py RENAMED
@@ -1,48 +1,431 @@
1
  """Plugwise Sensor component for Home Assistant."""
 
2
  from __future__ import annotations
3
 
4
- from homeassistant.components.sensor import SensorEntity
5
- from homeassistant.config_entries import ConfigEntry
6
- from homeassistant.core import HomeAssistant
7
- from homeassistant.helpers.entity_platform import AddEntitiesCallback
8
-
9
- from .const import (
10
- COORDINATOR, # pw-beta
11
- DOMAIN,
12
- LOGGER,
13
  )
14
- from .coordinator import PlugwiseDataUpdateCoordinator
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
15
  from .entity import PlugwiseEntity
16
- from .models import PW_SENSOR_TYPES, PlugwiseSensorEntityDescription
17
 
 
18
  PARALLEL_UPDATES = 0
19
 
20
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
21
  async def async_setup_entry(
22
  hass: HomeAssistant,
23
- config_entry: ConfigEntry,
24
- async_add_entities: AddEntitiesCallback,
25
  ) -> None:
26
  """Set up the Smile sensors from a config entry."""
27
- coordinator = hass.data[DOMAIN][config_entry.entry_id][COORDINATOR]
28
 
29
- entities: list[PlugwiseSensorEntity] = []
30
- for device_id, device in coordinator.data.devices.items():
31
- if not (sensors := device.get("sensors")):
32
- continue
33
- for description in PW_SENSOR_TYPES:
34
- if description.key not in sensors:
35
- continue
36
- entities.append(
37
- PlugwiseSensorEntity(
38
- coordinator,
39
- device_id,
40
- description,
41
- )
42
- )
43
- LOGGER.debug("Add %s sensor", description.key)
44
 
45
- async_add_entities(entities)
 
46
 
47
 
48
  class PlugwiseSensorEntity(PlugwiseEntity, SensorEntity):
@@ -58,10 +441,10 @@
58
  ) -> None:
59
  """Initialise the sensor."""
60
  super().__init__(coordinator, device_id)
61
- self.entity_description = description
62
  self._attr_unique_id = f"{device_id}-{description.key}"
 
63
 
64
  @property
65
  def native_value(self) -> int | float:
66
  """Return the value reported by the sensor."""
67
- return self.entity_description.value_fn(self.device["sensors"])
 
1
  """Plugwise Sensor component for Home Assistant."""
2
+
3
  from __future__ import annotations
4
 
5
+ from dataclasses import dataclass
6
+
7
+ from plugwise.constants import SensorType
8
+
9
+ from homeassistant.components.sensor import (
10
+ SensorDeviceClass,
11
+ SensorEntity,
12
+ SensorEntityDescription,
13
+ SensorStateClass,
14
  )
15
+ from homeassistant.const import (
16
+ LIGHT_LUX,
17
+ PERCENTAGE,
18
+ EntityCategory,
19
+ UnitOfElectricPotential,
20
+ UnitOfEnergy,
21
+ UnitOfPower,
22
+ UnitOfPressure,
23
+ UnitOfTemperature,
24
+ UnitOfVolume,
25
+ UnitOfVolumeFlowRate,
26
+ )
27
+ from homeassistant.core import HomeAssistant, callback
28
+ from homeassistant.helpers.entity_platform import AddConfigEntryEntitiesCallback
29
+
30
+ from .coordinator import PlugwiseConfigEntry, PlugwiseDataUpdateCoordinator
31
  from .entity import PlugwiseEntity
 
32
 
33
+ # Coordinator is used to centralize the data updates
34
  PARALLEL_UPDATES = 0
35
 
36
 
37
+ @dataclass(frozen=True)
38
+ class PlugwiseSensorEntityDescription(SensorEntityDescription):
39
+ """Describes Plugwise sensor entity."""
40
+
41
+ key: SensorType
42
+
43
+
44
+ SENSORS: tuple[PlugwiseSensorEntityDescription, ...] = (
45
+ PlugwiseSensorEntityDescription(
46
+ key="setpoint",
47
+ translation_key="setpoint",
48
+ native_unit_of_measurement=UnitOfTemperature.CELSIUS,
49
+ device_class=SensorDeviceClass.TEMPERATURE,
50
+ state_class=SensorStateClass.MEASUREMENT,
51
+ entity_category=EntityCategory.DIAGNOSTIC,
52
+ ),
53
+ PlugwiseSensorEntityDescription(
54
+ key="setpoint_high",
55
+ translation_key="cooling_setpoint",
56
+ native_unit_of_measurement=UnitOfTemperature.CELSIUS,
57
+ device_class=SensorDeviceClass.TEMPERATURE,
58
+ state_class=SensorStateClass.MEASUREMENT,
59
+ entity_category=EntityCategory.DIAGNOSTIC,
60
+ ),
61
+ PlugwiseSensorEntityDescription(
62
+ key="setpoint_low",
63
+ translation_key="heating_setpoint",
64
+ native_unit_of_measurement=UnitOfTemperature.CELSIUS,
65
+ device_class=SensorDeviceClass.TEMPERATURE,
66
+ state_class=SensorStateClass.MEASUREMENT,
67
+ entity_category=EntityCategory.DIAGNOSTIC,
68
+ ),
69
+ PlugwiseSensorEntityDescription(
70
+ key="temperature",
71
+ native_unit_of_measurement=UnitOfTemperature.CELSIUS,
72
+ device_class=SensorDeviceClass.TEMPERATURE,
73
+ entity_category=EntityCategory.DIAGNOSTIC,
74
+ state_class=SensorStateClass.MEASUREMENT,
75
+ ),
76
+ PlugwiseSensorEntityDescription(
77
+ key="intended_boiler_temperature",
78
+ translation_key="intended_boiler_temperature",
79
+ native_unit_of_measurement=UnitOfTemperature.CELSIUS,
80
+ device_class=SensorDeviceClass.TEMPERATURE,
81
+ entity_category=EntityCategory.DIAGNOSTIC,
82
+ state_class=SensorStateClass.MEASUREMENT,
83
+ ),
84
+ PlugwiseSensorEntityDescription(
85
+ key="temperature_difference",
86
+ translation_key="temperature_difference",
87
+ native_unit_of_measurement=UnitOfTemperature.CELSIUS,
88
+ device_class=SensorDeviceClass.TEMPERATURE,
89
+ entity_category=EntityCategory.DIAGNOSTIC,
90
+ state_class=SensorStateClass.MEASUREMENT,
91
+ ),
92
+ PlugwiseSensorEntityDescription(
93
+ key="outdoor_temperature",
94
+ translation_key="outdoor_temperature",
95
+ native_unit_of_measurement=UnitOfTemperature.CELSIUS,
96
+ device_class=SensorDeviceClass.TEMPERATURE,
97
+ state_class=SensorStateClass.MEASUREMENT,
98
+ ),
99
+ PlugwiseSensorEntityDescription(
100
+ key="outdoor_air_temperature",
101
+ translation_key="outdoor_air_temperature",
102
+ native_unit_of_measurement=UnitOfTemperature.CELSIUS,
103
+ device_class=SensorDeviceClass.TEMPERATURE,
104
+ entity_category=EntityCategory.DIAGNOSTIC,
105
+ state_class=SensorStateClass.MEASUREMENT,
106
+ ),
107
+ PlugwiseSensorEntityDescription(
108
+ key="water_temperature",
109
+ translation_key="water_temperature",
110
+ native_unit_of_measurement=UnitOfTemperature.CELSIUS,
111
+ device_class=SensorDeviceClass.TEMPERATURE,
112
+ entity_category=EntityCategory.DIAGNOSTIC,
113
+ state_class=SensorStateClass.MEASUREMENT,
114
+ ),
115
+ PlugwiseSensorEntityDescription(
116
+ key="return_temperature",
117
+ translation_key="return_temperature",
118
+ native_unit_of_measurement=UnitOfTemperature.CELSIUS,
119
+ device_class=SensorDeviceClass.TEMPERATURE,
120
+ entity_category=EntityCategory.DIAGNOSTIC,
121
+ state_class=SensorStateClass.MEASUREMENT,
122
+ ),
123
+ PlugwiseSensorEntityDescription(
124
+ key="electricity_consumed",
125
+ translation_key="electricity_consumed",
126
+ native_unit_of_measurement=UnitOfPower.WATT,
127
+ device_class=SensorDeviceClass.POWER,
128
+ state_class=SensorStateClass.MEASUREMENT,
129
+ ),
130
+ PlugwiseSensorEntityDescription(
131
+ key="electricity_produced",
132
+ translation_key="electricity_produced",
133
+ native_unit_of_measurement=UnitOfPower.WATT,
134
+ device_class=SensorDeviceClass.POWER,
135
+ state_class=SensorStateClass.MEASUREMENT,
136
+ entity_registry_enabled_default=False,
137
+ ),
138
+ PlugwiseSensorEntityDescription(
139
+ key="electricity_consumed_interval",
140
+ translation_key="electricity_consumed_interval",
141
+ native_unit_of_measurement=UnitOfEnergy.WATT_HOUR,
142
+ device_class=SensorDeviceClass.ENERGY,
143
+ state_class=SensorStateClass.TOTAL,
144
+ ),
145
+ PlugwiseSensorEntityDescription(
146
+ key="electricity_consumed_peak_interval",
147
+ translation_key="electricity_consumed_peak_interval",
148
+ native_unit_of_measurement=UnitOfEnergy.WATT_HOUR,
149
+ device_class=SensorDeviceClass.ENERGY,
150
+ state_class=SensorStateClass.TOTAL,
151
+ ),
152
+ PlugwiseSensorEntityDescription(
153
+ key="electricity_consumed_off_peak_interval",
154
+ translation_key="electricity_consumed_off_peak_interval",
155
+ native_unit_of_measurement=UnitOfEnergy.WATT_HOUR,
156
+ device_class=SensorDeviceClass.ENERGY,
157
+ state_class=SensorStateClass.TOTAL,
158
+ ),
159
+ PlugwiseSensorEntityDescription(
160
+ key="electricity_produced_interval",
161
+ translation_key="electricity_produced_interval",
162
+ native_unit_of_measurement=UnitOfEnergy.WATT_HOUR,
163
+ device_class=SensorDeviceClass.ENERGY,
164
+ state_class=SensorStateClass.TOTAL,
165
+ entity_registry_enabled_default=False,
166
+ ),
167
+ PlugwiseSensorEntityDescription(
168
+ key="electricity_produced_peak_interval",
169
+ translation_key="electricity_produced_peak_interval",
170
+ native_unit_of_measurement=UnitOfEnergy.WATT_HOUR,
171
+ device_class=SensorDeviceClass.ENERGY,
172
+ state_class=SensorStateClass.TOTAL,
173
+ ),
174
+ PlugwiseSensorEntityDescription(
175
+ key="electricity_produced_off_peak_interval",
176
+ translation_key="electricity_produced_off_peak_interval",
177
+ native_unit_of_measurement=UnitOfEnergy.WATT_HOUR,
178
+ device_class=SensorDeviceClass.ENERGY,
179
+ state_class=SensorStateClass.TOTAL,
180
+ ),
181
+ PlugwiseSensorEntityDescription(
182
+ key="electricity_consumed_point",
183
+ translation_key="electricity_consumed_point",
184
+ device_class=SensorDeviceClass.POWER,
185
+ native_unit_of_measurement=UnitOfPower.WATT,
186
+ state_class=SensorStateClass.MEASUREMENT,
187
+ ),
188
+ PlugwiseSensorEntityDescription(
189
+ key="electricity_consumed_off_peak_point",
190
+ translation_key="electricity_consumed_off_peak_point",
191
+ native_unit_of_measurement=UnitOfPower.WATT,
192
+ device_class=SensorDeviceClass.POWER,
193
+ state_class=SensorStateClass.MEASUREMENT,
194
+ ),
195
+ PlugwiseSensorEntityDescription(
196
+ key="electricity_consumed_peak_point",
197
+ translation_key="electricity_consumed_peak_point",
198
+ native_unit_of_measurement=UnitOfPower.WATT,
199
+ device_class=SensorDeviceClass.POWER,
200
+ state_class=SensorStateClass.MEASUREMENT,
201
+ ),
202
+ PlugwiseSensorEntityDescription(
203
+ key="electricity_consumed_off_peak_cumulative",
204
+ translation_key="electricity_consumed_off_peak_cumulative",
205
+ native_unit_of_measurement=UnitOfEnergy.KILO_WATT_HOUR,
206
+ device_class=SensorDeviceClass.ENERGY,
207
+ state_class=SensorStateClass.TOTAL_INCREASING,
208
+ ),
209
+ PlugwiseSensorEntityDescription(
210
+ key="electricity_consumed_peak_cumulative",
211
+ translation_key="electricity_consumed_peak_cumulative",
212
+ native_unit_of_measurement=UnitOfEnergy.KILO_WATT_HOUR,
213
+ device_class=SensorDeviceClass.ENERGY,
214
+ state_class=SensorStateClass.TOTAL_INCREASING,
215
+ ),
216
+ PlugwiseSensorEntityDescription(
217
+ key="electricity_produced_point",
218
+ translation_key="electricity_produced_point",
219
+ device_class=SensorDeviceClass.POWER,
220
+ native_unit_of_measurement=UnitOfPower.WATT,
221
+ state_class=SensorStateClass.MEASUREMENT,
222
+ ),
223
+ PlugwiseSensorEntityDescription(
224
+ key="electricity_produced_off_peak_point",
225
+ translation_key="electricity_produced_off_peak_point",
226
+ native_unit_of_measurement=UnitOfPower.WATT,
227
+ device_class=SensorDeviceClass.POWER,
228
+ state_class=SensorStateClass.MEASUREMENT,
229
+ ),
230
+ PlugwiseSensorEntityDescription(
231
+ key="electricity_produced_peak_point",
232
+ translation_key="electricity_produced_peak_point",
233
+ native_unit_of_measurement=UnitOfPower.WATT,
234
+ device_class=SensorDeviceClass.POWER,
235
+ state_class=SensorStateClass.MEASUREMENT,
236
+ ),
237
+ PlugwiseSensorEntityDescription(
238
+ key="electricity_produced_off_peak_cumulative",
239
+ translation_key="electricity_produced_off_peak_cumulative",
240
+ native_unit_of_measurement=UnitOfEnergy.KILO_WATT_HOUR,
241
+ device_class=SensorDeviceClass.ENERGY,
242
+ state_class=SensorStateClass.TOTAL_INCREASING,
243
+ ),
244
+ PlugwiseSensorEntityDescription(
245
+ key="electricity_produced_peak_cumulative",
246
+ translation_key="electricity_produced_peak_cumulative",
247
+ native_unit_of_measurement=UnitOfEnergy.KILO_WATT_HOUR,
248
+ device_class=SensorDeviceClass.ENERGY,
249
+ state_class=SensorStateClass.TOTAL_INCREASING,
250
+ ),
251
+ PlugwiseSensorEntityDescription(
252
+ key="electricity_phase_one_consumed",
253
+ translation_key="electricity_phase_one_consumed",
254
+ device_class=SensorDeviceClass.POWER,
255
+ native_unit_of_measurement=UnitOfPower.WATT,
256
+ state_class=SensorStateClass.MEASUREMENT,
257
+ ),
258
+ PlugwiseSensorEntityDescription(
259
+ key="electricity_phase_two_consumed",
260
+ translation_key="electricity_phase_two_consumed",
261
+ device_class=SensorDeviceClass.POWER,
262
+ native_unit_of_measurement=UnitOfPower.WATT,
263
+ state_class=SensorStateClass.MEASUREMENT,
264
+ ),
265
+ PlugwiseSensorEntityDescription(
266
+ key="electricity_phase_three_consumed",
267
+ translation_key="electricity_phase_three_consumed",
268
+ device_class=SensorDeviceClass.POWER,
269
+ native_unit_of_measurement=UnitOfPower.WATT,
270
+ state_class=SensorStateClass.MEASUREMENT,
271
+ ),
272
+ PlugwiseSensorEntityDescription(
273
+ key="electricity_phase_one_produced",
274
+ translation_key="electricity_phase_one_produced",
275
+ device_class=SensorDeviceClass.POWER,
276
+ native_unit_of_measurement=UnitOfPower.WATT,
277
+ state_class=SensorStateClass.MEASUREMENT,
278
+ ),
279
+ PlugwiseSensorEntityDescription(
280
+ key="electricity_phase_two_produced",
281
+ translation_key="electricity_phase_two_produced",
282
+ device_class=SensorDeviceClass.POWER,
283
+ native_unit_of_measurement=UnitOfPower.WATT,
284
+ state_class=SensorStateClass.MEASUREMENT,
285
+ ),
286
+ PlugwiseSensorEntityDescription(
287
+ key="electricity_phase_three_produced",
288
+ translation_key="electricity_phase_three_produced",
289
+ device_class=SensorDeviceClass.POWER,
290
+ native_unit_of_measurement=UnitOfPower.WATT,
291
+ state_class=SensorStateClass.MEASUREMENT,
292
+ ),
293
+ PlugwiseSensorEntityDescription(
294
+ key="voltage_phase_one",
295
+ translation_key="voltage_phase_one",
296
+ device_class=SensorDeviceClass.VOLTAGE,
297
+ native_unit_of_measurement=UnitOfElectricPotential.VOLT,
298
+ state_class=SensorStateClass.MEASUREMENT,
299
+ entity_registry_enabled_default=False,
300
+ ),
301
+ PlugwiseSensorEntityDescription(
302
+ key="voltage_phase_two",
303
+ translation_key="voltage_phase_two",
304
+ device_class=SensorDeviceClass.VOLTAGE,
305
+ native_unit_of_measurement=UnitOfElectricPotential.VOLT,
306
+ state_class=SensorStateClass.MEASUREMENT,
307
+ entity_registry_enabled_default=False,
308
+ ),
309
+ PlugwiseSensorEntityDescription(
310
+ key="voltage_phase_three",
311
+ translation_key="voltage_phase_three",
312
+ device_class=SensorDeviceClass.VOLTAGE,
313
+ native_unit_of_measurement=UnitOfElectricPotential.VOLT,
314
+ state_class=SensorStateClass.MEASUREMENT,
315
+ entity_registry_enabled_default=False,
316
+ ),
317
+ PlugwiseSensorEntityDescription(
318
+ key="gas_consumed_interval",
319
+ translation_key="gas_consumed_interval",
320
+ native_unit_of_measurement=UnitOfVolumeFlowRate.CUBIC_METERS_PER_HOUR,
321
+ state_class=SensorStateClass.MEASUREMENT,
322
+ ),
323
+ PlugwiseSensorEntityDescription(
324
+ key="gas_consumed_cumulative",
325
+ translation_key="gas_consumed_cumulative",
326
+ native_unit_of_measurement=UnitOfVolume.CUBIC_METERS,
327
+ device_class=SensorDeviceClass.GAS,
328
+ state_class=SensorStateClass.TOTAL,
329
+ ),
330
+ PlugwiseSensorEntityDescription(
331
+ key="net_electricity_point",
332
+ translation_key="net_electricity_point",
333
+ native_unit_of_measurement=UnitOfPower.WATT,
334
+ device_class=SensorDeviceClass.POWER,
335
+ state_class=SensorStateClass.MEASUREMENT,
336
+ ),
337
+ PlugwiseSensorEntityDescription(
338
+ key="net_electricity_cumulative",
339
+ translation_key="net_electricity_cumulative",
340
+ native_unit_of_measurement=UnitOfEnergy.KILO_WATT_HOUR,
341
+ device_class=SensorDeviceClass.ENERGY,
342
+ state_class=SensorStateClass.TOTAL,
343
+ ),
344
+ PlugwiseSensorEntityDescription(
345
+ key="battery",
346
+ native_unit_of_measurement=PERCENTAGE,
347
+ device_class=SensorDeviceClass.BATTERY,
348
+ entity_category=EntityCategory.DIAGNOSTIC,
349
+ state_class=SensorStateClass.MEASUREMENT,
350
+ ),
351
+ PlugwiseSensorEntityDescription(
352
+ key="illuminance",
353
+ native_unit_of_measurement=LIGHT_LUX,
354
+ device_class=SensorDeviceClass.ILLUMINANCE,
355
+ state_class=SensorStateClass.MEASUREMENT,
356
+ entity_category=EntityCategory.DIAGNOSTIC,
357
+ ),
358
+ PlugwiseSensorEntityDescription(
359
+ key="modulation_level",
360
+ translation_key="modulation_level",
361
+ native_unit_of_measurement=PERCENTAGE,
362
+ entity_category=EntityCategory.DIAGNOSTIC,
363
+ state_class=SensorStateClass.MEASUREMENT,
364
+ ),
365
+ PlugwiseSensorEntityDescription(
366
+ key="valve_position",
367
+ translation_key="valve_position",
368
+ entity_category=EntityCategory.DIAGNOSTIC,
369
+ native_unit_of_measurement=PERCENTAGE,
370
+ state_class=SensorStateClass.MEASUREMENT,
371
+ ),
372
+ PlugwiseSensorEntityDescription(
373
+ key="water_pressure",
374
+ translation_key="water_pressure",
375
+ native_unit_of_measurement=UnitOfPressure.BAR,
376
+ device_class=SensorDeviceClass.PRESSURE,
377
+ entity_category=EntityCategory.DIAGNOSTIC,
378
+ state_class=SensorStateClass.MEASUREMENT,
379
+ ),
380
+ PlugwiseSensorEntityDescription(
381
+ key="humidity",
382
+ native_unit_of_measurement=PERCENTAGE,
383
+ device_class=SensorDeviceClass.HUMIDITY,
384
+ state_class=SensorStateClass.MEASUREMENT,
385
+ ),
386
+ PlugwiseSensorEntityDescription(
387
+ key="dhw_temperature",
388
+ translation_key="dhw_temperature",
389
+ native_unit_of_measurement=UnitOfTemperature.CELSIUS,
390
+ device_class=SensorDeviceClass.TEMPERATURE,
391
+ entity_category=EntityCategory.DIAGNOSTIC,
392
+ state_class=SensorStateClass.MEASUREMENT,
393
+ ),
394
+ PlugwiseSensorEntityDescription(
395
+ key="domestic_hot_water_setpoint",
396
+ translation_key="domestic_hot_water_setpoint",
397
+ native_unit_of_measurement=UnitOfTemperature.CELSIUS,
398
+ device_class=SensorDeviceClass.TEMPERATURE,
399
+ entity_category=EntityCategory.DIAGNOSTIC,
400
+ state_class=SensorStateClass.MEASUREMENT,
401
+ ),
402
+ )
403
+
404
+
405
  async def async_setup_entry(
406
  hass: HomeAssistant,
407
+ entry: PlugwiseConfigEntry,
408
+ async_add_entities: AddConfigEntryEntitiesCallback,
409
  ) -> None:
410
  """Set up the Smile sensors from a config entry."""
411
+ coordinator = entry.runtime_data
412
 
413
+ @callback
414
+ def _add_entities() -> None:
415
+ """Add Entities."""
416
+ if not coordinator.new_devices:
417
+ return
418
+
419
+ async_add_entities(
420
+ PlugwiseSensorEntity(coordinator, device_id, description)
421
+ for device_id in coordinator.new_devices
422
+ if (sensors := coordinator.data[device_id].get("sensors"))
423
+ for description in SENSORS
424
+ if description.key in sensors
425
+ )
 
 
426
 
427
+ _add_entities()
428
+ entry.async_on_unload(coordinator.async_add_listener(_add_entities))
429
 
430
 
431
  class PlugwiseSensorEntity(PlugwiseEntity, SensorEntity):
 
441
  ) -> None:
442
  """Initialise the sensor."""
443
  super().__init__(coordinator, device_id)
 
444
  self._attr_unique_id = f"{device_id}-{description.key}"
445
+ self.entity_description = description
446
 
447
  @property
448
  def native_value(self) -> int | float:
449
  """Return the value reported by the sensor."""
450
+ return self.device["sensors"][self.entity_description.key]
/home/runner/work/progress/progress/clones/beta/{beta/custom_components → ha-core/homeassistant/components}/plugwise/strings.json RENAMED
@@ -1,48 +1,47 @@
1
  {
2
- "options": {
3
  "step": {
4
- "none": {
5
- "title": "No Options available",
6
- "description": "This Integration does not provide any Options"
7
- },
8
- "init": {
9
- "description": "Adjust Smile/Stretch Options",
10
  "data": {
11
- "cooling_on": "Anna: cooling-mode is on",
12
- "scan_interval": "Scan Interval (seconds) *) beta-only option",
13
- "homekit_emulation": "Homekit emulation (i.e. on hvac_off => Away) *) beta-only option",
14
- "refresh_interval": "Frontend refresh-time (1.5 - 5 seconds) *) beta-only option"
 
 
15
  }
16
- }
17
- }
18
- },
19
- "config": {
20
- "step": {
21
  "user": {
22
- "title": "Connect to the Plugwise Adam/Smile/Stretch",
23
- "description": "Please enter:",
24
  "data": {
25
- "password": "ID",
26
- "username": "Username",
27
- "host": "IP-address",
28
- "port": "Port number"
 
 
 
 
 
 
29
  }
30
  }
31
  },
32
  "error": {
33
- "cannot_connect": "Failed to connect",
34
- "invalid_auth": "Authentication failed",
35
  "invalid_setup": "Add your Adam instead of your Anna, see the documentation",
36
- "network_down": "Plugwise Zigbee network is down",
37
- "network_timeout": "Network communication timeout",
38
  "response_error": "Invalid XML data, or error indication received",
39
- "stick_init": "Initialization of Plugwise USB-stick failed",
40
- "unknown": "Unknown error!",
41
  "unsupported": "Device with unsupported firmware"
42
  },
43
  "abort": {
44
- "already_configured": "This device is already configured",
45
- "anna_with_adam": "Both Anna and Adam detected. Add your Adam instead of your Anna"
 
 
46
  }
47
  },
48
  "entity": {
@@ -60,29 +59,40 @@
60
  "name": "Flame state"
61
  },
62
  "heating_state": {
63
- "name": "Heating"
64
  },
65
  "cooling_state": {
66
- "name": "Cooling"
67
  },
68
- "slave_boiler_state": {
69
  "name": "Secondary boiler state"
70
  },
71
  "plugwise_notification": {
72
  "name": "Plugwise notification"
73
  }
74
  },
 
 
 
 
 
75
  "climate": {
76
  "plugwise": {
77
  "state_attributes": {
 
 
 
78
  "preset_mode": {
79
  "state": {
80
  "asleep": "Night",
81
- "away": "Away",
82
- "home": "Home",
83
  "no_frost": "Anti-frost",
84
  "vacation": "Vacation"
85
  }
 
 
 
86
  }
87
  }
88
  }
@@ -93,16 +103,27 @@
93
  },
94
  "max_dhw_temperature": {
95
  "name": "Domestic hot water setpoint"
 
 
 
96
  }
97
  },
98
  "select": {
99
  "dhw_mode": {
100
  "name": "DHW mode",
101
  "state": {
102
- "auto": "Auto",
103
- "boost": "Boost",
104
- "comfort": "Comfort",
105
- "off": "Off"
 
 
 
 
 
 
 
 
106
  }
107
  },
108
  "regulation_mode": {
@@ -110,13 +131,16 @@
110
  "state": {
111
  "bleeding_cold": "Bleeding cold",
112
  "bleeding_hot": "Bleeding hot",
113
- "cooling": "Cooling",
114
- "heating": "Heating",
115
- "off": "Off"
116
  }
117
  },
118
- "thermostat_schedule": {
119
- "name": "Thermostat schedule"
 
 
 
120
  }
121
  },
122
  "sensor": {
@@ -127,10 +151,7 @@
127
  "name": "Cooling setpoint"
128
  },
129
  "heating_setpoint": {
130
- "name": " Heating setpoint"
131
- },
132
- "temperature": {
133
- "name": "Temperature"
134
  },
135
  "intended_boiler_temperature": {
136
  "name": "Intended boiler temperature"
@@ -156,12 +177,6 @@
156
  "electricity_produced": {
157
  "name": "Electricity produced"
158
  },
159
- "electricity_consumed_point": {
160
- "name": "Electricity consumed point"
161
- },
162
- "electricity_produced_point": {
163
- "name": "Electricity produced point"
164
- },
165
  "electricity_consumed_interval": {
166
  "name": "Electricity consumed interval"
167
  },
@@ -169,7 +184,7 @@
169
  "name": "Electricity consumed peak interval"
170
  },
171
  "electricity_consumed_off_peak_interval": {
172
- "name": "Electricity consumed off peak interval"
173
  },
174
  "electricity_produced_interval": {
175
  "name": "Electricity produced interval"
@@ -178,28 +193,34 @@
178
  "name": "Electricity produced peak interval"
179
  },
180
  "electricity_produced_off_peak_interval": {
181
- "name": "Electricity produced off peak interval"
 
 
 
182
  },
183
  "electricity_consumed_off_peak_point": {
184
- "name": "Electricity consumed off peak point"
185
  },
186
  "electricity_consumed_peak_point": {
187
  "name": "Electricity consumed peak point"
188
  },
189
  "electricity_consumed_off_peak_cumulative": {
190
- "name": "Electricity consumed off peak cumulative"
191
  },
192
  "electricity_consumed_peak_cumulative": {
193
  "name": "Electricity consumed peak cumulative"
194
  },
 
 
 
195
  "electricity_produced_off_peak_point": {
196
- "name": "Electricity produced off peak point"
197
  },
198
  "electricity_produced_peak_point": {
199
  "name": "Electricity produced peak point"
200
  },
201
  "electricity_produced_off_peak_cumulative": {
202
- "name": "Electricity produced off peak_cumulative"
203
  },
204
  "electricity_produced_peak_cumulative": {
205
  "name": "Electricity produced peak cumulative"
@@ -243,12 +264,6 @@
243
  "net_electricity_cumulative": {
244
  "name": "Net electricity cumulative"
245
  },
246
- "battery": {
247
- "name": "Battery"
248
- },
249
- "illuminance": {
250
- "name": "Illuminance"
251
- },
252
  "modulation_level": {
253
  "name": "Modulation level"
254
  },
@@ -258,9 +273,6 @@
258
  "water_pressure": {
259
  "name": "Water pressure"
260
  },
261
- "relative_humidity": {
262
- "name": "Relative humidity"
263
- },
264
  "dhw_temperature": {
265
  "name": "DHW temperature"
266
  },
@@ -268,22 +280,45 @@
268
  "name": "DHW setpoint"
269
  },
270
  "maximum_boiler_temperature": {
271
- "name": "Maximum boiler temperature setpoint"
272
  }
273
  },
274
  "switch": {
275
  "cooling_ena_switch": {
276
- "name": "Cooling"
277
  },
278
  "dhw_cm_switch": {
279
  "name": "DHW comfort mode"
280
  },
281
  "lock": {
282
- "name": "Lock"
283
  },
284
  "relay": {
285
  "name": "Relay"
286
  }
287
  }
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
288
  }
289
  }
 
1
  {
2
+ "config": {
3
  "step": {
4
+ "reconfigure": {
5
+ "description": "Update configuration for {title}.",
 
 
 
 
6
  "data": {
7
+ "host": "[%key:common::config_flow::data::ip%]",
8
+ "port": "[%key:common::config_flow::data::port%]"
9
+ },
10
+ "data_description": {
11
+ "host": "[%key:component::plugwise::config::step::user::data_description::host%]",
12
+ "port": "[%key:component::plugwise::config::step::user::data_description::port%]"
13
  }
14
+ },
 
 
 
 
15
  "user": {
16
+ "title": "Connect to the Smile",
17
+ "description": "Please enter",
18
  "data": {
19
+ "host": "[%key:common::config_flow::data::ip%]",
20
+ "password": "Smile ID",
21
+ "port": "[%key:common::config_flow::data::port%]",
22
+ "username": "Smile username"
23
+ },
24
+ "data_description": {
25
+ "password": "The Smile ID printed on the label on the back of your Adam, Smile-T, or P1.",
26
+ "host": "The hostname or IP-address of your Smile. You can find it in your router or the Plugwise app.",
27
+ "port": "By default your Smile uses port 80, normally you should not have to change this.",
28
+ "username": "Default is `smile`, or `stretch` for the legacy Stretch."
29
  }
30
  }
31
  },
32
  "error": {
33
+ "cannot_connect": "[%key:common::config_flow::error::cannot_connect%]",
34
+ "invalid_auth": "[%key:common::config_flow::error::invalid_auth%]",
35
  "invalid_setup": "Add your Adam instead of your Anna, see the documentation",
 
 
36
  "response_error": "Invalid XML data, or error indication received",
37
+ "unknown": "[%key:common::config_flow::error::unknown%]",
 
38
  "unsupported": "Device with unsupported firmware"
39
  },
40
  "abort": {
41
+ "already_configured": "[%key:common::config_flow::abort::already_configured_service%]",
42
+ "anna_with_adam": "Both Anna and Adam detected. Add your Adam instead of your Anna",
43
+ "not_the_same_smile": "The configured Smile ID does not match the Smile ID on the requested IP address.",
44
+ "reconfigure_successful": "[%key:common::config_flow::abort::reconfigure_successful%]"
45
  }
46
  },
47
  "entity": {
 
59
  "name": "Flame state"
60
  },
61
  "heating_state": {
62
+ "name": "[%key:component::climate::entity_component::_::state_attributes::hvac_action::state::heating%]"
63
  },
64
  "cooling_state": {
65
+ "name": "[%key:component::climate::entity_component::_::state_attributes::hvac_action::state::cooling%]"
66
  },
67
+ "secondary_boiler_state": {
68
  "name": "Secondary boiler state"
69
  },
70
  "plugwise_notification": {
71
  "name": "Plugwise notification"
72
  }
73
  },
74
+ "button": {
75
+ "reboot": {
76
+ "name": "Reboot"
77
+ }
78
+ },
79
  "climate": {
80
  "plugwise": {
81
  "state_attributes": {
82
+ "available_schemas": {
83
+ "name": "Available schemas"
84
+ },
85
  "preset_mode": {
86
  "state": {
87
  "asleep": "Night",
88
+ "away": "[%key:common::state::not_home%]",
89
+ "home": "[%key:common::state::home%]",
90
  "no_frost": "Anti-frost",
91
  "vacation": "Vacation"
92
  }
93
+ },
94
+ "selected_schema": {
95
+ "name": "Selected schema"
96
  }
97
  }
98
  }
 
103
  },
104
  "max_dhw_temperature": {
105
  "name": "Domestic hot water setpoint"
106
+ },
107
+ "temperature_offset": {
108
+ "name": "Temperature offset"
109
  }
110
  },
111
  "select": {
112
  "dhw_mode": {
113
  "name": "DHW mode",
114
  "state": {
115
+ "off": "[%key:common::state::off%]",
116
+ "auto": "[%key:common::state::auto%]",
117
+ "boost": "[%key:component::climate::entity_component::_::state_attributes::preset_mode::state::boost%]",
118
+ "comfort": "[%key:component::climate::entity_component::_::state_attributes::preset_mode::state::comfort%]"
119
+ }
120
+ },
121
+ "gateway_mode": {
122
+ "name": "Gateway mode",
123
+ "state": {
124
+ "away": "Pause",
125
+ "full": "[%key:common::state::normal%]",
126
+ "vacation": "Vacation"
127
  }
128
  },
129
  "regulation_mode": {
 
131
  "state": {
132
  "bleeding_cold": "Bleeding cold",
133
  "bleeding_hot": "Bleeding hot",
134
+ "cooling": "[%key:component::climate::entity_component::_::state_attributes::hvac_action::state::cooling%]",
135
+ "heating": "[%key:component::climate::entity_component::_::state_attributes::hvac_action::state::heating%]",
136
+ "off": "[%key:common::state::off%]"
137
  }
138
  },
139
+ "select_schedule": {
140
+ "name": "Thermostat schedule",
141
+ "state": {
142
+ "off": "[%key:common::state::off%]"
143
+ }
144
  }
145
  },
146
  "sensor": {
 
151
  "name": "Cooling setpoint"
152
  },
153
  "heating_setpoint": {
154
+ "name": "Heating setpoint"
 
 
 
155
  },
156
  "intended_boiler_temperature": {
157
  "name": "Intended boiler temperature"
 
177
  "electricity_produced": {
178
  "name": "Electricity produced"
179
  },
 
 
 
 
 
 
180
  "electricity_consumed_interval": {
181
  "name": "Electricity consumed interval"
182
  },
 
184
  "name": "Electricity consumed peak interval"
185
  },
186
  "electricity_consumed_off_peak_interval": {
187
+ "name": "Electricity consumed off-peak interval"
188
  },
189
  "electricity_produced_interval": {
190
  "name": "Electricity produced interval"
 
193
  "name": "Electricity produced peak interval"
194
  },
195
  "electricity_produced_off_peak_interval": {
196
+ "name": "Electricity produced off-peak interval"
197
+ },
198
+ "electricity_consumed_point": {
199
+ "name": "Electricity consumed point"
200
  },
201
  "electricity_consumed_off_peak_point": {
202
+ "name": "Electricity consumed off-peak point"
203
  },
204
  "electricity_consumed_peak_point": {
205
  "name": "Electricity consumed peak point"
206
  },
207
  "electricity_consumed_off_peak_cumulative": {
208
+ "name": "Electricity consumed off-peak cumulative"
209
  },
210
  "electricity_consumed_peak_cumulative": {
211
  "name": "Electricity consumed peak cumulative"
212
  },
213
+ "electricity_produced_point": {
214
+ "name": "Electricity produced point"
215
+ },
216
  "electricity_produced_off_peak_point": {
217
+ "name": "Electricity produced off-peak point"
218
  },
219
  "electricity_produced_peak_point": {
220
  "name": "Electricity produced peak point"
221
  },
222
  "electricity_produced_off_peak_cumulative": {
223
+ "name": "Electricity produced off-peak cumulative"
224
  },
225
  "electricity_produced_peak_cumulative": {
226
  "name": "Electricity produced peak cumulative"
 
264
  "net_electricity_cumulative": {
265
  "name": "Net electricity cumulative"
266
  },
 
 
 
 
 
 
267
  "modulation_level": {
268
  "name": "Modulation level"
269
  },
 
273
  "water_pressure": {
274
  "name": "Water pressure"
275
  },
 
 
 
276
  "dhw_temperature": {
277
  "name": "DHW temperature"
278
  },
 
280
  "name": "DHW setpoint"
281
  },
282
  "maximum_boiler_temperature": {
283
+ "name": "Maximum boiler temperature"
284
  }
285
  },
286
  "switch": {
287
  "cooling_ena_switch": {
288
+ "name": "[%key:component::climate::entity_component::_::state_attributes::hvac_action::state::cooling%]"
289
  },
290
  "dhw_cm_switch": {
291
  "name": "DHW comfort mode"
292
  },
293
  "lock": {
294
+ "name": "[%key:component::lock::title%]"
295
  },
296
  "relay": {
297
  "name": "Relay"
298
  }
299
  }
300
+ },
301
+ "exceptions": {
302
+ "authentication_failed": {
303
+ "message": "[%key:common::config_flow::error::invalid_auth%]"
304
+ },
305
+ "data_incomplete_or_missing": {
306
+ "message": "Data incomplete or missing."
307
+ },
308
+ "error_communicating_with_api": {
309
+ "message": "Error communicating with API: {error}."
310
+ },
311
+ "failed_to_connect": {
312
+ "message": "[%key:common::config_flow::error::cannot_connect%]"
313
+ },
314
+ "invalid_xml_data": {
315
+ "message": "[%key:component::plugwise::config::error::response_error%]"
316
+ },
317
+ "unsupported_firmware": {
318
+ "message": "[%key:component::plugwise::config::error::unsupported%]"
319
+ },
320
+ "unsupported_hvac_mode_requested": {
321
+ "message": "Unsupported mode {hvac_mode} requested, valid modes are: {hvac_modes}."
322
+ }
323
  }
324
  }
/home/runner/work/progress/progress/clones/beta/{beta/custom_components → ha-core/homeassistant/components}/plugwise/switch.py RENAMED
@@ -1,42 +1,83 @@
1
  """Plugwise Switch component for HomeAssistant."""
 
2
  from __future__ import annotations
3
 
 
4
  from typing import Any
5
 
6
- from homeassistant.components.switch import SwitchEntity
7
- from homeassistant.config_entries import ConfigEntry
8
- from homeassistant.core import HomeAssistant
9
- from homeassistant.helpers.entity_platform import AddEntitiesCallback
10
-
11
- from .const import (
12
- COORDINATOR, # pw-beta
13
- DOMAIN,
14
- LOGGER,
15
  )
16
- from .coordinator import PlugwiseDataUpdateCoordinator
 
 
 
 
17
  from .entity import PlugwiseEntity
18
- from .models import PW_SWITCH_TYPES, PlugwiseSwitchEntityDescription
19
  from .util import plugwise_command
20
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
21
 
22
  async def async_setup_entry(
23
  hass: HomeAssistant,
24
- config_entry: ConfigEntry,
25
- async_add_entities: AddEntitiesCallback,
26
  ) -> None:
27
  """Set up the Smile switches from a config entry."""
28
- coordinator = hass.data[DOMAIN][config_entry.entry_id][COORDINATOR]
29
- entities: list[PlugwiseSwitchEntity] = []
30
- for device_id, device in coordinator.data.devices.items():
31
- if not (switches := device.get("switches")):
32
- continue
33
- for description in PW_SWITCH_TYPES:
34
- if description.key not in switches:
35
- continue
36
- entities.append(PlugwiseSwitchEntity(coordinator, device_id, description))
37
- LOGGER.debug("Add %s switch", description.key)
 
 
 
 
 
38
 
39
- async_add_entities(entities)
 
40
 
41
 
42
  class PlugwiseSwitchEntity(PlugwiseEntity, SwitchEntity):
@@ -52,13 +93,13 @@
52
  ) -> None:
53
  """Set up the Plugwise API."""
54
  super().__init__(coordinator, device_id)
55
- self.entity_description = description
56
  self._attr_unique_id = f"{device_id}-{description.key}"
 
57
 
58
  @property
59
  def is_on(self) -> bool:
60
  """Return True if entity is on."""
61
- return self.entity_description.value_fn(self.device["switches"])
62
 
63
  @plugwise_command
64
  async def async_turn_on(self, **kwargs: Any) -> None:
 
1
  """Plugwise Switch component for HomeAssistant."""
2
+
3
  from __future__ import annotations
4
 
5
+ from dataclasses import dataclass
6
  from typing import Any
7
 
8
+ from plugwise.constants import SwitchType
9
+
10
+ from homeassistant.components.switch import (
11
+ SwitchDeviceClass,
12
+ SwitchEntity,
13
+ SwitchEntityDescription,
 
 
 
14
  )
15
+ from homeassistant.const import EntityCategory
16
+ from homeassistant.core import HomeAssistant, callback
17
+ from homeassistant.helpers.entity_platform import AddConfigEntryEntitiesCallback
18
+
19
+ from .coordinator import PlugwiseConfigEntry, PlugwiseDataUpdateCoordinator
20
  from .entity import PlugwiseEntity
 
21
  from .util import plugwise_command
22
 
23
+ PARALLEL_UPDATES = 0
24
+
25
+
26
+ @dataclass(frozen=True)
27
+ class PlugwiseSwitchEntityDescription(SwitchEntityDescription):
28
+ """Describes Plugwise switch entity."""
29
+
30
+ key: SwitchType
31
+
32
+
33
+ SWITCHES: tuple[PlugwiseSwitchEntityDescription, ...] = (
34
+ PlugwiseSwitchEntityDescription(
35
+ key="dhw_cm_switch",
36
+ translation_key="dhw_cm_switch",
37
+ entity_category=EntityCategory.CONFIG,
38
+ ),
39
+ PlugwiseSwitchEntityDescription(
40
+ key="lock",
41
+ translation_key="lock",
42
+ entity_category=EntityCategory.CONFIG,
43
+ ),
44
+ PlugwiseSwitchEntityDescription(
45
+ key="relay",
46
+ translation_key="relay",
47
+ device_class=SwitchDeviceClass.SWITCH,
48
+ ),
49
+ PlugwiseSwitchEntityDescription(
50
+ key="cooling_ena_switch",
51
+ translation_key="cooling_ena_switch",
52
+ entity_category=EntityCategory.CONFIG,
53
+ ),
54
+ )
55
+
56
 
57
  async def async_setup_entry(
58
  hass: HomeAssistant,
59
+ entry: PlugwiseConfigEntry,
60
+ async_add_entities: AddConfigEntryEntitiesCallback,
61
  ) -> None:
62
  """Set up the Smile switches from a config entry."""
63
+ coordinator = entry.runtime_data
64
+
65
+ @callback
66
+ def _add_entities() -> None:
67
+ """Add Entities."""
68
+ if not coordinator.new_devices:
69
+ return
70
+
71
+ async_add_entities(
72
+ PlugwiseSwitchEntity(coordinator, device_id, description)
73
+ for device_id in coordinator.new_devices
74
+ if (switches := coordinator.data[device_id].get("switches"))
75
+ for description in SWITCHES
76
+ if description.key in switches
77
+ )
78
 
79
+ _add_entities()
80
+ entry.async_on_unload(coordinator.async_add_listener(_add_entities))
81
 
82
 
83
  class PlugwiseSwitchEntity(PlugwiseEntity, SwitchEntity):
 
93
  ) -> None:
94
  """Set up the Plugwise API."""
95
  super().__init__(coordinator, device_id)
 
96
  self._attr_unique_id = f"{device_id}-{description.key}"
97
+ self.entity_description = description
98
 
99
  @property
100
  def is_on(self) -> bool:
101
  """Return True if entity is on."""
102
+ return self.device["switches"][self.entity_description.key]
103
 
104
  @plugwise_command
105
  async def async_turn_on(self, **kwargs: Any) -> None:
/home/runner/work/progress/progress/clones/beta/{beta/custom_components → ha-core/homeassistant/components}/plugwise/util.py RENAMED
@@ -1,21 +1,18 @@
1
  """Utilities for Plugwise."""
 
2
  from collections.abc import Awaitable, Callable, Coroutine
3
- from typing import Any, Concatenate, TypeVar
4
 
5
- from typing_extensions import ParamSpec
6
 
7
  from homeassistant.exceptions import HomeAssistantError
8
- from plugwise.exceptions import PlugwiseException
9
 
 
10
  from .entity import PlugwiseEntity
11
 
12
- _PlugwiseEntityT = TypeVar("_PlugwiseEntityT", bound=PlugwiseEntity)
13
- _R = TypeVar("_R")
14
- _P = ParamSpec("_P")
15
-
16
 
17
- def plugwise_command(
18
- func: Callable[Concatenate[_PlugwiseEntityT, _P], Awaitable[_R]]
19
  ) -> Callable[Concatenate[_PlugwiseEntityT, _P], Coroutine[Any, Any, _R]]:
20
  """Decorate Plugwise calls that send commands/make changes to the device.
21
 
@@ -28,10 +25,14 @@
28
  ) -> _R:
29
  try:
30
  return await func(self, *args, **kwargs)
31
- except PlugwiseException as error:
32
  raise HomeAssistantError(
33
- f"Error communicating with API: {error}"
34
- ) from error
 
 
 
 
35
  finally:
36
  await self.coordinator.async_request_refresh()
37
 
 
1
  """Utilities for Plugwise."""
2
+
3
  from collections.abc import Awaitable, Callable, Coroutine
4
+ from typing import Any, Concatenate
5
 
6
+ from plugwise.exceptions import PlugwiseException
7
 
8
  from homeassistant.exceptions import HomeAssistantError
 
9
 
10
+ from .const import DOMAIN
11
  from .entity import PlugwiseEntity
12
 
 
 
 
 
13
 
14
+ def plugwise_command[_PlugwiseEntityT: PlugwiseEntity, **_P, _R](
15
+ func: Callable[Concatenate[_PlugwiseEntityT, _P], Awaitable[_R]],
16
  ) -> Callable[Concatenate[_PlugwiseEntityT, _P], Coroutine[Any, Any, _R]]:
17
  """Decorate Plugwise calls that send commands/make changes to the device.
18
 
 
25
  ) -> _R:
26
  try:
27
  return await func(self, *args, **kwargs)
28
+ except PlugwiseException as err:
29
  raise HomeAssistantError(
30
+ translation_domain=DOMAIN,
31
+ translation_key="error_communicating_with_api",
32
+ translation_placeholders={
33
+ "error": str(err),
34
+ },
35
+ ) from err
36
  finally:
37
  await self.coordinator.async_request_refresh()
38