Skip to content

Commit 1bef707

Browse files
authored
Add support for switchbot climate panel (home-assistant#155124)
1 parent 2125a41 commit 1bef707

File tree

5 files changed

+97
-1
lines changed

5 files changed

+97
-1
lines changed

homeassistant/components/switchbot/__init__.py

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -101,6 +101,7 @@
101101
SupportedModels.PLUG_MINI_EU.value: [Platform.SWITCH, Platform.SENSOR],
102102
SupportedModels.RELAY_SWITCH_2PM.value: [Platform.SWITCH, Platform.SENSOR],
103103
SupportedModels.GARAGE_DOOR_OPENER.value: [Platform.COVER, Platform.SENSOR],
104+
SupportedModels.CLIMATE_PANEL.value: [Platform.SENSOR, Platform.BINARY_SENSOR],
104105
}
105106
CLASS_BY_DEVICE = {
106107
SupportedModels.CEILING_LIGHT.value: switchbot.SwitchbotCeilingLight,

homeassistant/components/switchbot/binary_sensor.py

Lines changed: 0 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,6 @@
2424
),
2525
"motion_detected": BinarySensorEntityDescription(
2626
key="pir_state",
27-
name=None,
2827
device_class=BinarySensorDeviceClass.MOTION,
2928
),
3029
"contact_open": BinarySensorEntityDescription(

homeassistant/components/switchbot/const.py

Lines changed: 3 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -57,6 +57,7 @@ class SupportedModels(StrEnum):
5757
RELAY_SWITCH_2PM = "relay_switch_2pm"
5858
K11_PLUS_VACUUM = "k11+_vacuum"
5959
GARAGE_DOOR_OPENER = "garage_door_opener"
60+
CLIMATE_PANEL = "climate_panel"
6061

6162

6263
CONNECTABLE_SUPPORTED_MODEL_TYPES = {
@@ -93,6 +94,7 @@ class SupportedModels(StrEnum):
9394
SwitchbotModel.RELAY_SWITCH_2PM: SupportedModels.RELAY_SWITCH_2PM,
9495
SwitchbotModel.K11_VACUUM: SupportedModels.K11_PLUS_VACUUM,
9596
SwitchbotModel.GARAGE_DOOR_OPENER: SupportedModels.GARAGE_DOOR_OPENER,
97+
SwitchbotModel.CLIMATE_PANEL: SupportedModels.CLIMATE_PANEL,
9698
}
9799

98100
NON_CONNECTABLE_SUPPORTED_MODEL_TYPES = {
@@ -106,6 +108,7 @@ class SupportedModels(StrEnum):
106108
SwitchbotModel.REMOTE: SupportedModels.REMOTE,
107109
SwitchbotModel.HUBMINI_MATTER: SupportedModels.HUBMINI_MATTER,
108110
SwitchbotModel.HUB3: SupportedModels.HUB3,
111+
SwitchbotModel.CLIMATE_PANEL: SupportedModels.CLIMATE_PANEL,
109112
}
110113

111114
SUPPORTED_MODEL_TYPES = (

tests/components/switchbot/__init__.py

Lines changed: 29 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -1171,3 +1171,32 @@ def make_advertisement(
11711171
connectable=True,
11721172
tx_power=-127,
11731173
)
1174+
1175+
1176+
CLIMATE_PANEL_SERVICE_INFO = BluetoothServiceInfoBleak(
1177+
name="Climate Panel",
1178+
manufacturer_data={
1179+
2409: b"\xb0\xe9\xfe\x8e\x98Oi_\x06\x9a,\x00\x00\x00\x00\xe4\x00\x08\x04\x00\x01\x00\x00"
1180+
},
1181+
service_data={
1182+
"0000fd3d-0000-1000-8000-00805f9b34fb": b"\x00 _\x00\x10\xf3\xd8@",
1183+
},
1184+
service_uuids=["cba20d00-224d-11e6-9fb8-0002a5d5c51b"],
1185+
address="AA:BB:CC:DD:EE:FF",
1186+
rssi=-60,
1187+
source="local",
1188+
advertisement=generate_advertisement_data(
1189+
local_name="Climate Panel",
1190+
manufacturer_data={
1191+
2409: b"\xb0\xe9\xfe\x8e\x98Oi_\x06\x9a,\x00\x00\x00\x00\xe4\x00\x08\x04\x00\x01\x00\x00"
1192+
},
1193+
service_data={
1194+
"0000fd3d-0000-1000-8000-00805f9b34fb": b"\x00 _\x00\x10\xf3\xd8@"
1195+
},
1196+
service_uuids=["cba20d00-224d-11e6-9fb8-0002a5d5c51b"],
1197+
),
1198+
device=generate_ble_device("AA:BB:CC:DD:EE:FF", "Climate Panel"),
1199+
time=0,
1200+
connectable=True,
1201+
tx_power=-127,
1202+
)

tests/components/switchbot/test_sensor.py

Lines changed: 64 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,7 @@
2424

2525
from . import (
2626
CIRCULATOR_FAN_SERVICE_INFO,
27+
CLIMATE_PANEL_SERVICE_INFO,
2728
EVAPORATIVE_HUMIDIFIER_SERVICE_INFO,
2829
HUB3_SERVICE_INFO,
2930
HUBMINI_MATTER_SERVICE_INFO,
@@ -728,3 +729,66 @@ async def test_relay_switch_2pm_sensor(hass: HomeAssistant) -> None:
728729

729730
assert await hass.config_entries.async_unload(entry.entry_id)
730731
await hass.async_block_till_done()
732+
733+
734+
@pytest.mark.usefixtures("entity_registry_enabled_by_default")
735+
async def test_climate_panel_sensor(hass: HomeAssistant) -> None:
736+
"""Test setting up creates the sensor for Climate Panel."""
737+
await async_setup_component(hass, DOMAIN, {})
738+
inject_bluetooth_service_info(hass, CLIMATE_PANEL_SERVICE_INFO)
739+
740+
entry = MockConfigEntry(
741+
domain=DOMAIN,
742+
data={
743+
CONF_ADDRESS: "AA:BB:CC:DD:EE:FF",
744+
CONF_NAME: "test-name",
745+
CONF_SENSOR_TYPE: "climate_panel",
746+
},
747+
unique_id="aabbccddeeff",
748+
)
749+
entry.add_to_hass(hass)
750+
751+
assert await hass.config_entries.async_setup(entry.entry_id)
752+
await hass.async_block_till_done()
753+
754+
assert len(hass.states.async_all("sensor")) == 4
755+
assert len(hass.states.async_all("binary_sensor")) == 2
756+
757+
temperature_sensor = hass.states.get("sensor.test_name_temperature")
758+
temperature_sensor_attrs = temperature_sensor.attributes
759+
assert temperature_sensor.state == "26.6"
760+
assert temperature_sensor_attrs[ATTR_FRIENDLY_NAME] == "test-name Temperature"
761+
assert temperature_sensor_attrs[ATTR_UNIT_OF_MEASUREMENT] == "°C"
762+
assert temperature_sensor_attrs[ATTR_STATE_CLASS] == "measurement"
763+
764+
humidity_sensor = hass.states.get("sensor.test_name_humidity")
765+
humidity_sensor_attrs = humidity_sensor.attributes
766+
assert humidity_sensor.state == "44"
767+
assert humidity_sensor_attrs[ATTR_FRIENDLY_NAME] == "test-name Humidity"
768+
assert humidity_sensor_attrs[ATTR_UNIT_OF_MEASUREMENT] == "%"
769+
assert humidity_sensor_attrs[ATTR_STATE_CLASS] == "measurement"
770+
771+
rssi_sensor = hass.states.get("sensor.test_name_bluetooth_signal")
772+
rssi_sensor_attrs = rssi_sensor.attributes
773+
assert rssi_sensor.state == "-60"
774+
assert rssi_sensor_attrs[ATTR_FRIENDLY_NAME] == "test-name Bluetooth signal"
775+
assert rssi_sensor_attrs[ATTR_UNIT_OF_MEASUREMENT] == "dBm"
776+
777+
battery_sensor = hass.states.get("sensor.test_name_battery")
778+
battery_sensor_attrs = battery_sensor.attributes
779+
assert battery_sensor.state == "95"
780+
assert battery_sensor_attrs[ATTR_FRIENDLY_NAME] == "test-name Battery"
781+
assert battery_sensor_attrs[ATTR_STATE_CLASS] == "measurement"
782+
783+
light_sensor = hass.states.get("binary_sensor.test_name_light")
784+
light_sensor_attrs = light_sensor.attributes
785+
assert light_sensor.state == "off"
786+
assert light_sensor_attrs[ATTR_FRIENDLY_NAME] == "test-name Light"
787+
788+
motion_sensor = hass.states.get("binary_sensor.test_name_motion")
789+
motion_sensor_attrs = motion_sensor.attributes
790+
assert motion_sensor.state == "on"
791+
assert motion_sensor_attrs[ATTR_FRIENDLY_NAME] == "test-name Motion"
792+
793+
assert await hass.config_entries.async_unload(entry.entry_id)
794+
await hass.async_block_till_done()

0 commit comments

Comments
 (0)