From e70af204eec1a67a293edbb23213a530bfeee7a2 Mon Sep 17 00:00:00 2001 From: Marc Mueller <30130371+cdce8p@users.noreply.github.com> Date: Mon, 1 Jan 2024 18:56:44 +0100 Subject: [PATCH 01/26] Enable strict typing for airthings_ble (#106815) --- .strict-typing | 1 + homeassistant/components/airthings_ble/__init__.py | 6 +++--- mypy.ini | 10 ++++++++++ 3 files changed, 14 insertions(+), 3 deletions(-) diff --git a/.strict-typing b/.strict-typing index aa9c801fbf6799..1bc3308f533ccf 100644 --- a/.strict-typing +++ b/.strict-typing @@ -49,6 +49,7 @@ homeassistant.components.aftership.* homeassistant.components.air_quality.* homeassistant.components.airly.* homeassistant.components.airnow.* +homeassistant.components.airthings_ble.* homeassistant.components.airvisual.* homeassistant.components.airvisual_pro.* homeassistant.components.airzone.* diff --git a/homeassistant/components/airthings_ble/__init__.py b/homeassistant/components/airthings_ble/__init__.py index d7e6bddbcd4e10..c642ebf95635c3 100644 --- a/homeassistant/components/airthings_ble/__init__.py +++ b/homeassistant/components/airthings_ble/__init__.py @@ -4,7 +4,7 @@ from datetime import timedelta import logging -from airthings_ble import AirthingsBluetoothDeviceData +from airthings_ble import AirthingsBluetoothDeviceData, AirthingsDevice from homeassistant.components import bluetooth from homeassistant.config_entries import ConfigEntry @@ -37,13 +37,13 @@ async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool: f"Could not find Airthings device with address {address}" ) - async def _async_update_method(): + async def _async_update_method() -> AirthingsDevice: """Get data from Airthings BLE.""" ble_device = bluetooth.async_ble_device_from_address(hass, address) airthings = AirthingsBluetoothDeviceData(_LOGGER, elevation, is_metric) try: - data = await airthings.update_device(ble_device) + data = await airthings.update_device(ble_device) # type: ignore[arg-type] except Exception as err: raise UpdateFailed(f"Unable to fetch data: {err}") from err diff --git a/mypy.ini b/mypy.ini index e19c6c6fa925d7..e4546526722ec5 100644 --- a/mypy.ini +++ b/mypy.ini @@ -250,6 +250,16 @@ disallow_untyped_defs = true warn_return_any = true warn_unreachable = true +[mypy-homeassistant.components.airthings_ble.*] +check_untyped_defs = true +disallow_incomplete_defs = true +disallow_subclassing_any = true +disallow_untyped_calls = true +disallow_untyped_decorators = true +disallow_untyped_defs = true +warn_return_any = true +warn_unreachable = true + [mypy-homeassistant.components.airvisual.*] check_untyped_defs = true disallow_incomplete_defs = true From 5a0997bac0e9d8501a270cfb59245ae8ff0767dd Mon Sep 17 00:00:00 2001 From: Joe Neuman Date: Tue, 2 Jan 2024 15:19:00 -0800 Subject: [PATCH 02/26] Fix qBittorrent torrent count when empty (#106903) * Fix qbittorrent torrent cound when empty * lint fix * Change based on comment --- homeassistant/components/qbittorrent/sensor.py | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/homeassistant/components/qbittorrent/sensor.py b/homeassistant/components/qbittorrent/sensor.py index 9373aec85444df..78e8ba59d44eed 100644 --- a/homeassistant/components/qbittorrent/sensor.py +++ b/homeassistant/components/qbittorrent/sensor.py @@ -165,6 +165,10 @@ def count_torrents_in_states( coordinator: QBittorrentDataCoordinator, states: list[str] ) -> int: """Count the number of torrents in specified states.""" + # When torrents are not in the returned data, there are none, return 0. + if "torrents" not in coordinator.data: + return 0 + if not states: return len(coordinator.data["torrents"]) From d87baba96f8fe228f35ca4ae86fd186651989ace Mon Sep 17 00:00:00 2001 From: Patrick Frazer Date: Wed, 3 Jan 2024 15:10:00 -0500 Subject: [PATCH 03/26] Bump dropmqttapi to 1.0.2 (#106978) --- homeassistant/components/drop_connect/manifest.json | 2 +- requirements_all.txt | 2 +- requirements_test_all.txt | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/homeassistant/components/drop_connect/manifest.json b/homeassistant/components/drop_connect/manifest.json index f65c1848aff007..5df34fce561c99 100644 --- a/homeassistant/components/drop_connect/manifest.json +++ b/homeassistant/components/drop_connect/manifest.json @@ -7,5 +7,5 @@ "documentation": "https://www.home-assistant.io/integrations/drop_connect", "iot_class": "local_push", "mqtt": ["drop_connect/discovery/#"], - "requirements": ["dropmqttapi==1.0.1"] + "requirements": ["dropmqttapi==1.0.2"] } diff --git a/requirements_all.txt b/requirements_all.txt index 46b89f491a99d2..8b874569d668e3 100644 --- a/requirements_all.txt +++ b/requirements_all.txt @@ -716,7 +716,7 @@ dovado==0.4.1 dremel3dpy==2.1.1 # homeassistant.components.drop_connect -dropmqttapi==1.0.1 +dropmqttapi==1.0.2 # homeassistant.components.dsmr dsmr-parser==1.3.1 diff --git a/requirements_test_all.txt b/requirements_test_all.txt index ee1a9b2ac35fad..767511cc9583fa 100644 --- a/requirements_test_all.txt +++ b/requirements_test_all.txt @@ -585,7 +585,7 @@ discovery30303==0.2.1 dremel3dpy==2.1.1 # homeassistant.components.drop_connect -dropmqttapi==1.0.1 +dropmqttapi==1.0.2 # homeassistant.components.dsmr dsmr-parser==1.3.1 From 1c94a94ba2a4c25e771aded846b15752a5c6320e Mon Sep 17 00:00:00 2001 From: Sid <27780930+autinerd@users.noreply.github.com> Date: Thu, 28 Dec 2023 21:59:56 +0100 Subject: [PATCH 04/26] bump openwebifpy to 4.0.3 (#106593) --- homeassistant/components/enigma2/manifest.json | 2 +- requirements_all.txt | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/homeassistant/components/enigma2/manifest.json b/homeassistant/components/enigma2/manifest.json index 7909db3b7c7fc2..19a2cf863f915a 100644 --- a/homeassistant/components/enigma2/manifest.json +++ b/homeassistant/components/enigma2/manifest.json @@ -5,5 +5,5 @@ "documentation": "https://www.home-assistant.io/integrations/enigma2", "iot_class": "local_polling", "loggers": ["openwebif"], - "requirements": ["openwebifpy==4.0.2"] + "requirements": ["openwebifpy==4.0.3"] } diff --git a/requirements_all.txt b/requirements_all.txt index 8b874569d668e3..88c2f0e85068d1 100644 --- a/requirements_all.txt +++ b/requirements_all.txt @@ -1425,7 +1425,7 @@ openhomedevice==2.2.0 opensensemap-api==0.2.0 # homeassistant.components.enigma2 -openwebifpy==4.0.2 +openwebifpy==4.0.3 # homeassistant.components.luci openwrt-luci-rpc==1.1.16 From aef129afafbe2f9aa2f3e3c441af6b1490337a92 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?St=C3=A5le=20Stor=C3=B8=20Hauknes?= Date: Wed, 3 Jan 2024 21:08:58 +0100 Subject: [PATCH 05/26] Close stale connections (Airthings BLE) (#106748) Co-authored-by: J. Nick Koston --- homeassistant/components/airthings_ble/__init__.py | 3 +++ 1 file changed, 3 insertions(+) diff --git a/homeassistant/components/airthings_ble/__init__.py b/homeassistant/components/airthings_ble/__init__.py index c642ebf95635c3..1d62442f14de1d 100644 --- a/homeassistant/components/airthings_ble/__init__.py +++ b/homeassistant/components/airthings_ble/__init__.py @@ -5,6 +5,7 @@ import logging from airthings_ble import AirthingsBluetoothDeviceData, AirthingsDevice +from bleak_retry_connector import close_stale_connections_by_address from homeassistant.components import bluetooth from homeassistant.config_entries import ConfigEntry @@ -30,6 +31,8 @@ async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool: is_metric = hass.config.units is METRIC_SYSTEM assert address is not None + await close_stale_connections_by_address(address) + ble_device = bluetooth.async_ble_device_from_address(hass, address) if not ble_device: From ce5455fefc2a7efa83ede07fa4e7366aa60d584c Mon Sep 17 00:00:00 2001 From: Sid <27780930+autinerd@users.noreply.github.com> Date: Thu, 4 Jan 2024 01:40:59 +0100 Subject: [PATCH 06/26] Bump openwebifpy to 4.0.4 (#107000) --- homeassistant/components/enigma2/manifest.json | 2 +- requirements_all.txt | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/homeassistant/components/enigma2/manifest.json b/homeassistant/components/enigma2/manifest.json index 19a2cf863f915a..42fbcb5b9bc605 100644 --- a/homeassistant/components/enigma2/manifest.json +++ b/homeassistant/components/enigma2/manifest.json @@ -5,5 +5,5 @@ "documentation": "https://www.home-assistant.io/integrations/enigma2", "iot_class": "local_polling", "loggers": ["openwebif"], - "requirements": ["openwebifpy==4.0.3"] + "requirements": ["openwebifpy==4.0.4"] } diff --git a/requirements_all.txt b/requirements_all.txt index 88c2f0e85068d1..11d12f195a8607 100644 --- a/requirements_all.txt +++ b/requirements_all.txt @@ -1425,7 +1425,7 @@ openhomedevice==2.2.0 opensensemap-api==0.2.0 # homeassistant.components.enigma2 -openwebifpy==4.0.3 +openwebifpy==4.0.4 # homeassistant.components.luci openwrt-luci-rpc==1.1.16 From 8c9875c3cc24a34807253669c3d6170aeb6ebe23 Mon Sep 17 00:00:00 2001 From: Maciej Bieniek Date: Thu, 4 Jan 2024 01:45:59 +0100 Subject: [PATCH 07/26] Get Shelly RPC device `gen` from config entry data (#107019) Use gen from config entry data --- homeassistant/components/shelly/config_flow.py | 17 +++++++++-------- homeassistant/components/shelly/const.py | 2 ++ homeassistant/components/shelly/coordinator.py | 3 ++- homeassistant/components/shelly/utils.py | 5 +++-- 4 files changed, 16 insertions(+), 11 deletions(-) diff --git a/homeassistant/components/shelly/config_flow.py b/homeassistant/components/shelly/config_flow.py index 68b0f1f8cccb18..29daf05016370e 100644 --- a/homeassistant/components/shelly/config_flow.py +++ b/homeassistant/components/shelly/config_flow.py @@ -25,6 +25,7 @@ from .const import ( CONF_BLE_SCANNER_MODE, + CONF_GEN, CONF_SLEEP_PERIOD, DOMAIN, LOGGER, @@ -84,7 +85,7 @@ async def validate_input( "title": rpc_device.name, CONF_SLEEP_PERIOD: sleep_period, "model": rpc_device.shelly.get("model"), - "gen": gen, + CONF_GEN: gen, } # Gen1 @@ -99,7 +100,7 @@ async def validate_input( "title": block_device.name, CONF_SLEEP_PERIOD: get_block_device_sleep_period(block_device.settings), "model": block_device.model, - "gen": gen, + CONF_GEN: gen, } @@ -153,7 +154,7 @@ async def async_step_user( **user_input, CONF_SLEEP_PERIOD: device_info[CONF_SLEEP_PERIOD], "model": device_info["model"], - "gen": device_info["gen"], + CONF_GEN: device_info[CONF_GEN], }, ) errors["base"] = "firmware_not_fully_provisioned" @@ -190,7 +191,7 @@ async def async_step_credentials( CONF_HOST: self.host, CONF_SLEEP_PERIOD: device_info[CONF_SLEEP_PERIOD], "model": device_info["model"], - "gen": device_info["gen"], + CONF_GEN: device_info[CONF_GEN], }, ) errors["base"] = "firmware_not_fully_provisioned" @@ -288,7 +289,7 @@ async def async_step_confirm_discovery( "host": self.host, CONF_SLEEP_PERIOD: self.device_info[CONF_SLEEP_PERIOD], "model": self.device_info["model"], - "gen": self.device_info["gen"], + CONF_GEN: self.device_info[CONF_GEN], }, ) self._set_confirm_only() @@ -321,7 +322,7 @@ async def async_step_reauth_confirm( except (DeviceConnectionError, InvalidAuthError, FirmwareUnsupported): return self.async_abort(reason="reauth_unsuccessful") - if self.entry.data.get("gen", 1) != 1: + if self.entry.data.get(CONF_GEN, 1) != 1: user_input[CONF_USERNAME] = "admin" try: await validate_input(self.hass, host, info, user_input) @@ -334,7 +335,7 @@ async def async_step_reauth_confirm( await self.hass.config_entries.async_reload(self.entry.entry_id) return self.async_abort(reason="reauth_successful") - if self.entry.data.get("gen", 1) in BLOCK_GENERATIONS: + if self.entry.data.get(CONF_GEN, 1) in BLOCK_GENERATIONS: schema = { vol.Required(CONF_USERNAME): str, vol.Required(CONF_PASSWORD): str, @@ -363,7 +364,7 @@ def async_get_options_flow(config_entry: ConfigEntry) -> OptionsFlowHandler: def async_supports_options_flow(cls, config_entry: ConfigEntry) -> bool: """Return options flow support for this handler.""" return ( - config_entry.data.get("gen") in RPC_GENERATIONS + config_entry.data.get(CONF_GEN) in RPC_GENERATIONS and not config_entry.data.get(CONF_SLEEP_PERIOD) and config_entry.data.get("model") != MODEL_WALL_DISPLAY ) diff --git a/homeassistant/components/shelly/const.py b/homeassistant/components/shelly/const.py index ca1c450c9faaab..1e2c22691fb6c9 100644 --- a/homeassistant/components/shelly/const.py +++ b/homeassistant/components/shelly/const.py @@ -214,3 +214,5 @@ class BLEScannerMode(StrEnum): MODEL_MOTION_2, MODEL_VALVE, ) + +CONF_GEN = "gen" diff --git a/homeassistant/components/shelly/coordinator.py b/homeassistant/components/shelly/coordinator.py index a7659ecc392f9b..77fa0bd2efd166 100644 --- a/homeassistant/components/shelly/coordinator.py +++ b/homeassistant/components/shelly/coordinator.py @@ -33,6 +33,7 @@ ATTR_GENERATION, BATTERY_DEVICES_WITH_PERMANENT_CONNECTION, CONF_BLE_SCANNER_MODE, + CONF_GEN, CONF_SLEEP_PERIOD, DATA_CONFIG_ENTRY, DOMAIN, @@ -135,7 +136,7 @@ def async_setup(self) -> None: manufacturer="Shelly", model=aioshelly.const.MODEL_NAMES.get(self.model, self.model), sw_version=self.sw_version, - hw_version=f"gen{self.device.gen} ({self.model})", + hw_version=f"gen{self.entry.data[CONF_GEN]} ({self.model})", configuration_url=f"http://{self.entry.data[CONF_HOST]}", ) self.device_id = device_entry.id diff --git a/homeassistant/components/shelly/utils.py b/homeassistant/components/shelly/utils.py index a43d9cb0bcb07d..d40b22ca50a4fa 100644 --- a/homeassistant/components/shelly/utils.py +++ b/homeassistant/components/shelly/utils.py @@ -34,6 +34,7 @@ from .const import ( BASIC_INPUTS_EVENTS_TYPES, CONF_COAP_PORT, + CONF_GEN, DEFAULT_COAP_PORT, DEVICES_WITHOUT_FIRMWARE_CHANGELOG, DOMAIN, @@ -281,7 +282,7 @@ def get_info_auth(info: dict[str, Any]) -> bool: def get_info_gen(info: dict[str, Any]) -> int: """Return the device generation from shelly info.""" - return int(info.get("gen", 1)) + return int(info.get(CONF_GEN, 1)) def get_model_name(info: dict[str, Any]) -> str: @@ -325,7 +326,7 @@ def get_rpc_entity_name( def get_device_entry_gen(entry: ConfigEntry) -> int: """Return the device generation from config entry.""" - return entry.data.get("gen", 1) + return entry.data.get(CONF_GEN, 1) def get_rpc_key_instances(keys_dict: dict[str, Any], key: str) -> list[str]: From 427077a4c9b52c3845381c356067b953c30b8ab7 Mon Sep 17 00:00:00 2001 From: "J. Nick Koston" Date: Wed, 3 Jan 2024 14:41:50 -1000 Subject: [PATCH 08/26] Fix missing backwards compatiblity layer for humidifier supported_features (#107026) fixes #107018 --- homeassistant/components/humidifier/__init__.py | 2 +- tests/components/humidifier/test_init.py | 5 +++++ 2 files changed, 6 insertions(+), 1 deletion(-) diff --git a/homeassistant/components/humidifier/__init__.py b/homeassistant/components/humidifier/__init__.py index 75d4f0fd22507e..0d8f2e29561640 100644 --- a/homeassistant/components/humidifier/__init__.py +++ b/homeassistant/components/humidifier/__init__.py @@ -214,7 +214,7 @@ def state_attributes(self) -> dict[str, Any]: if self.target_humidity is not None: data[ATTR_HUMIDITY] = self.target_humidity - if HumidifierEntityFeature.MODES in self.supported_features: + if HumidifierEntityFeature.MODES in self.supported_features_compat: data[ATTR_MODE] = self.mode return data diff --git a/tests/components/humidifier/test_init.py b/tests/components/humidifier/test_init.py index 45da5ba750f7f7..24cf4b6d962939 100644 --- a/tests/components/humidifier/test_init.py +++ b/tests/components/humidifier/test_init.py @@ -7,6 +7,7 @@ from homeassistant.components import humidifier from homeassistant.components.humidifier import ( + ATTR_MODE, HumidifierEntity, HumidifierEntityFeature, ) @@ -75,6 +76,8 @@ def test_deprecated_supported_features_ints(caplog: pytest.LogCaptureFixture) -> """Test deprecated supported features ints.""" class MockHumidifierEntity(HumidifierEntity): + _attr_mode = "mode1" + @property def supported_features(self) -> int: """Return supported features.""" @@ -89,3 +92,5 @@ def supported_features(self) -> int: caplog.clear() assert entity.supported_features_compat is HumidifierEntityFeature(1) assert "is using deprecated supported features values" not in caplog.text + + assert entity.state_attributes[ATTR_MODE] == "mode1" From e2acc70128ec070a8d6f42e490ed2c23fad9ff78 Mon Sep 17 00:00:00 2001 From: Joost Lekkerkerker Date: Thu, 4 Jan 2024 09:11:44 +0100 Subject: [PATCH 09/26] Use async_register in streamlabswater (#107060) --- homeassistant/components/streamlabswater/__init__.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/homeassistant/components/streamlabswater/__init__.py b/homeassistant/components/streamlabswater/__init__.py index 986b5de8049e22..82e8777a7e1ae5 100644 --- a/homeassistant/components/streamlabswater/__init__.py +++ b/homeassistant/components/streamlabswater/__init__.py @@ -112,7 +112,7 @@ def set_away_mode(service: ServiceCall) -> None: ) client.update_location(location_id, away_mode) - hass.services.register( + hass.services.async_register( DOMAIN, SERVICE_SET_AWAY_MODE, set_away_mode, schema=SET_AWAY_MODE_SCHEMA ) From 5529a85a2bdf60bdeaced2d0c95110c03865cdaa Mon Sep 17 00:00:00 2001 From: Joost Lekkerkerker Date: Thu, 4 Jan 2024 09:13:34 +0100 Subject: [PATCH 10/26] Fix data access in streamlabs water (#107062) * Fix data access in streamlabs water * Fix data access in streamlabs water --- homeassistant/components/streamlabswater/coordinator.py | 2 +- homeassistant/components/streamlabswater/sensor.py | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/homeassistant/components/streamlabswater/coordinator.py b/homeassistant/components/streamlabswater/coordinator.py index a11eced5a6e5c7..dc57ae78810a52 100644 --- a/homeassistant/components/streamlabswater/coordinator.py +++ b/homeassistant/components/streamlabswater/coordinator.py @@ -44,7 +44,7 @@ async def _async_update_data(self) -> dict[str, StreamlabsData]: def _update_data(self) -> dict[str, StreamlabsData]: locations = self.client.get_locations() res = {} - for location in locations: + for location in locations["locations"]: location_id = location["locationId"] water_usage = self.client.get_water_usage_summary(location_id) res[location_id] = StreamlabsData( diff --git a/homeassistant/components/streamlabswater/sensor.py b/homeassistant/components/streamlabswater/sensor.py index 0b249b7c4e54b7..6c869a6d1bc959 100644 --- a/homeassistant/components/streamlabswater/sensor.py +++ b/homeassistant/components/streamlabswater/sensor.py @@ -27,7 +27,7 @@ async def async_setup_entry( entities = [] - for location_id in coordinator.data.values(): + for location_id in coordinator.data: entities.extend( [ StreamLabsDailyUsage(coordinator, location_id), From 80b45edb2e64632d95ad1712559639ac4ee00577 Mon Sep 17 00:00:00 2001 From: Robert Resch Date: Fri, 5 Jan 2024 10:53:59 +0100 Subject: [PATCH 11/26] Fix mobile_app cloudhook creation (#107068) --- homeassistant/components/cloud/__init__.py | 17 +++++ .../components/mobile_app/__init__.py | 13 ++-- .../components/mobile_app/http_api.py | 5 +- homeassistant/components/mobile_app/util.py | 20 ++++++ tests/components/cloud/conftest.py | 1 + tests/components/cloud/test_init.py | 64 ++++++++++++++++++- tests/components/mobile_app/test_init.py | 10 +-- 7 files changed, 113 insertions(+), 17 deletions(-) diff --git a/homeassistant/components/cloud/__init__.py b/homeassistant/components/cloud/__init__.py index d7d57835e3ade1..6e5cddd0f28764 100644 --- a/homeassistant/components/cloud/__init__.py +++ b/homeassistant/components/cloud/__init__.py @@ -5,6 +5,7 @@ from collections.abc import Awaitable, Callable from datetime import datetime, timedelta from enum import Enum +from typing import cast from hass_nabucasa import Cloud import voluptuous as vol @@ -176,6 +177,22 @@ def async_active_subscription(hass: HomeAssistant) -> bool: return async_is_logged_in(hass) and not hass.data[DOMAIN].subscription_expired +async def async_get_or_create_cloudhook(hass: HomeAssistant, webhook_id: str) -> str: + """Get or create a cloudhook.""" + if not async_is_connected(hass): + raise CloudNotConnected + + if not async_is_logged_in(hass): + raise CloudNotAvailable + + cloud: Cloud[CloudClient] = hass.data[DOMAIN] + cloudhooks = cloud.client.cloudhooks + if hook := cloudhooks.get(webhook_id): + return cast(str, hook["cloudhook_url"]) + + return await async_create_cloudhook(hass, webhook_id) + + @bind_hass async def async_create_cloudhook(hass: HomeAssistant, webhook_id: str) -> str: """Create a cloudhook.""" diff --git a/homeassistant/components/mobile_app/__init__.py b/homeassistant/components/mobile_app/__init__.py index cb5c0ae5c3ddd2..124ef750baa128 100644 --- a/homeassistant/components/mobile_app/__init__.py +++ b/homeassistant/components/mobile_app/__init__.py @@ -36,6 +36,7 @@ ) from .helpers import savable_state from .http_api import RegistrationsView +from .util import async_create_cloud_hook from .webhook import handle_webhook PLATFORMS = [Platform.SENSOR, Platform.BINARY_SENSOR, Platform.DEVICE_TRACKER] @@ -103,26 +104,20 @@ async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool: registration_name = f"Mobile App: {registration[ATTR_DEVICE_NAME]}" webhook_register(hass, DOMAIN, registration_name, webhook_id, handle_webhook) - async def create_cloud_hook() -> None: - """Create a cloud hook.""" - hook = await cloud.async_create_cloudhook(hass, webhook_id) - hass.config_entries.async_update_entry( - entry, data={**entry.data, CONF_CLOUDHOOK_URL: hook} - ) - async def manage_cloudhook(state: cloud.CloudConnectionState) -> None: if ( state is cloud.CloudConnectionState.CLOUD_CONNECTED and CONF_CLOUDHOOK_URL not in entry.data ): - await create_cloud_hook() + await async_create_cloud_hook(hass, webhook_id, entry) if ( CONF_CLOUDHOOK_URL not in entry.data and cloud.async_active_subscription(hass) and cloud.async_is_connected(hass) ): - await create_cloud_hook() + await async_create_cloud_hook(hass, webhook_id, entry) + entry.async_on_unload(cloud.async_listen_connection_change(hass, manage_cloudhook)) await hass.config_entries.async_forward_entry_setups(entry, PLATFORMS) diff --git a/homeassistant/components/mobile_app/http_api.py b/homeassistant/components/mobile_app/http_api.py index 3c34a291df15e9..92bb473d51acdb 100644 --- a/homeassistant/components/mobile_app/http_api.py +++ b/homeassistant/components/mobile_app/http_api.py @@ -35,6 +35,7 @@ SCHEMA_APP_DATA, ) from .helpers import supports_encryption +from .util import async_create_cloud_hook class RegistrationsView(HomeAssistantView): @@ -69,8 +70,8 @@ async def post(self, request: Request, data: dict) -> Response: webhook_id = secrets.token_hex() if cloud.async_active_subscription(hass): - data[CONF_CLOUDHOOK_URL] = await cloud.async_create_cloudhook( - hass, webhook_id + data[CONF_CLOUDHOOK_URL] = await async_create_cloud_hook( + hass, webhook_id, None ) data[CONF_WEBHOOK_ID] = webhook_id diff --git a/homeassistant/components/mobile_app/util.py b/homeassistant/components/mobile_app/util.py index 45641861e5cd61..a7871d935edf37 100644 --- a/homeassistant/components/mobile_app/util.py +++ b/homeassistant/components/mobile_app/util.py @@ -1,8 +1,11 @@ """Mobile app utility functions.""" from __future__ import annotations +import asyncio from typing import TYPE_CHECKING +from homeassistant.components import cloud +from homeassistant.config_entries import ConfigEntry from homeassistant.core import HomeAssistant, callback from .const import ( @@ -10,6 +13,7 @@ ATTR_PUSH_TOKEN, ATTR_PUSH_URL, ATTR_PUSH_WEBSOCKET_CHANNEL, + CONF_CLOUDHOOK_URL, DATA_CONFIG_ENTRIES, DATA_DEVICES, DATA_NOTIFY, @@ -53,3 +57,19 @@ def get_notify_service(hass: HomeAssistant, webhook_id: str) -> str | None: return target_service return None + + +_CLOUD_HOOK_LOCK = asyncio.Lock() + + +async def async_create_cloud_hook( + hass: HomeAssistant, webhook_id: str, entry: ConfigEntry | None +) -> str: + """Create a cloud hook.""" + async with _CLOUD_HOOK_LOCK: + hook = await cloud.async_get_or_create_cloudhook(hass, webhook_id) + if entry: + hass.config_entries.async_update_entry( + entry, data={**entry.data, CONF_CLOUDHOOK_URL: hook} + ) + return hook diff --git a/tests/components/cloud/conftest.py b/tests/components/cloud/conftest.py index ef8cb037cdbf72..42852b15206110 100644 --- a/tests/components/cloud/conftest.py +++ b/tests/components/cloud/conftest.py @@ -109,6 +109,7 @@ def mock_is_connected() -> bool: is_connected = PropertyMock(side_effect=mock_is_connected) type(mock_cloud).is_connected = is_connected + type(mock_cloud.iot).connected = is_connected # Properties that we mock as attributes. mock_cloud.expiration_date = utcnow() diff --git a/tests/components/cloud/test_init.py b/tests/components/cloud/test_init.py index e12775d5a4a5f4..850f8e12e02085 100644 --- a/tests/components/cloud/test_init.py +++ b/tests/components/cloud/test_init.py @@ -1,12 +1,18 @@ """Test the cloud component.""" +from collections.abc import Callable, Coroutine from typing import Any -from unittest.mock import patch +from unittest.mock import MagicMock, patch from hass_nabucasa import Cloud import pytest from homeassistant.components import cloud -from homeassistant.components.cloud.const import DOMAIN +from homeassistant.components.cloud import ( + CloudNotAvailable, + CloudNotConnected, + async_get_or_create_cloudhook, +) +from homeassistant.components.cloud.const import DOMAIN, PREF_CLOUDHOOKS from homeassistant.components.cloud.prefs import STORAGE_KEY from homeassistant.const import EVENT_HOMEASSISTANT_STOP from homeassistant.core import Context, HomeAssistant @@ -214,3 +220,57 @@ async def test_remote_ui_url(hass: HomeAssistant, mock_cloud_fixture) -> None: cl.client.prefs._prefs["remote_domain"] = "example.com" assert cloud.async_remote_ui_url(hass) == "https://example.com" + + +async def test_async_get_or_create_cloudhook( + hass: HomeAssistant, + cloud: MagicMock, + set_cloud_prefs: Callable[[dict[str, Any]], Coroutine[Any, Any, None]], +) -> None: + """Test async_get_or_create_cloudhook.""" + assert await async_setup_component(hass, "cloud", {"cloud": {}}) + await hass.async_block_till_done() + + webhook_id = "mock-webhook-id" + cloudhook_url = "https://cloudhook.nabu.casa/abcdefg" + + with patch( + "homeassistant.components.cloud.async_create_cloudhook", + return_value=cloudhook_url, + ) as async_create_cloudhook_mock: + # create cloudhook as it does not exist + assert (await async_get_or_create_cloudhook(hass, webhook_id)) == cloudhook_url + async_create_cloudhook_mock.assert_called_once_with(hass, webhook_id) + + await set_cloud_prefs( + { + PREF_CLOUDHOOKS: { + webhook_id: { + "webhook_id": webhook_id, + "cloudhook_id": "random-id", + "cloudhook_url": cloudhook_url, + "managed": True, + } + } + } + ) + + async_create_cloudhook_mock.reset_mock() + + # get cloudhook as it exists + assert await async_get_or_create_cloudhook(hass, webhook_id) == cloudhook_url + async_create_cloudhook_mock.assert_not_called() + + # Simulate logged out + cloud.id_token = None + + # Not logged in + with pytest.raises(CloudNotAvailable): + await async_get_or_create_cloudhook(hass, webhook_id) + + # Simulate disconnected + cloud.iot.state = "disconnected" + + # Not connected + with pytest.raises(CloudNotConnected): + await async_get_or_create_cloudhook(hass, webhook_id) diff --git a/tests/components/mobile_app/test_init.py b/tests/components/mobile_app/test_init.py index d504703c222339..6a365e84fb0f1f 100644 --- a/tests/components/mobile_app/test_init.py +++ b/tests/components/mobile_app/test_init.py @@ -88,15 +88,17 @@ async def _test_create_cloud_hook( ), patch( "homeassistant.components.cloud.async_is_connected", return_value=True ), patch( - "homeassistant.components.cloud.async_create_cloudhook", autospec=True - ) as mock_create_cloudhook: + "homeassistant.components.cloud.async_get_or_create_cloudhook", autospec=True + ) as mock_async_get_or_create_cloudhook: cloud_hook = "https://hook-url" - mock_create_cloudhook.return_value = cloud_hook + mock_async_get_or_create_cloudhook.return_value = cloud_hook assert await hass.config_entries.async_setup(config_entry.entry_id) await hass.async_block_till_done() assert config_entry.state is ConfigEntryState.LOADED - await additional_steps(config_entry, mock_create_cloudhook, cloud_hook) + await additional_steps( + config_entry, mock_async_get_or_create_cloudhook, cloud_hook + ) async def test_create_cloud_hook_on_setup( From c56d118e8b9622c0c57d2f3be43e78e3b04ba7ca Mon Sep 17 00:00:00 2001 From: Erik Montnemery Date: Thu, 4 Jan 2024 13:25:09 +0100 Subject: [PATCH 12/26] Deduplicate handling of duplicated constants (#107074) * Deduplicate handling of duplicated constants * Use DeprecatedConstant + DeprecatedConstantEnum * Fixup * Remove test cases with unnamed tuples --- homeassistant/const.py | 383 +++++++++++++++------------ homeassistant/core.py | 46 +--- homeassistant/helpers/deprecation.py | 25 +- tests/helpers/test_deprecation.py | 32 --- 4 files changed, 236 insertions(+), 250 deletions(-) diff --git a/homeassistant/const.py b/homeassistant/const.py index 6afa0430ba36f3..a6927aa8165aed 100644 --- a/homeassistant/const.py +++ b/homeassistant/const.py @@ -2,7 +2,15 @@ from __future__ import annotations from enum import StrEnum -from typing import Any, Final +from functools import partial +from typing import Final + +from .helpers.deprecation import ( + DeprecatedConstant, + DeprecatedConstantEnum, + check_if_deprecated_constant, + dir_with_deprecated_constants, +) APPLICATION_NAME: Final = "HomeAssistant" MAJOR_VERSION: Final = 2024 @@ -307,146 +315,135 @@ class Platform(StrEnum): # #### DEVICE CLASSES #### # DEVICE_CLASS_* below are deprecated as of 2021.12 # use the SensorDeviceClass enum instead. -_DEPRECATED_DEVICE_CLASS_AQI: Final = ("aqi", "SensorDeviceClass.AQI", "2025.1") -_DEPRECATED_DEVICE_CLASS_BATTERY: Final = ( +_DEPRECATED_DEVICE_CLASS_AQI: Final = DeprecatedConstant( + "aqi", "SensorDeviceClass.AQI", "2025.1" +) +_DEPRECATED_DEVICE_CLASS_BATTERY: Final = DeprecatedConstant( "battery", "SensorDeviceClass.BATTERY", "2025.1", ) -_DEPRECATED_DEVICE_CLASS_CO: Final = ( +_DEPRECATED_DEVICE_CLASS_CO: Final = DeprecatedConstant( "carbon_monoxide", "SensorDeviceClass.CO", "2025.1", ) -_DEPRECATED_DEVICE_CLASS_CO2: Final = ( +_DEPRECATED_DEVICE_CLASS_CO2: Final = DeprecatedConstant( "carbon_dioxide", "SensorDeviceClass.CO2", "2025.1", ) -_DEPRECATED_DEVICE_CLASS_CURRENT: Final = ( +_DEPRECATED_DEVICE_CLASS_CURRENT: Final = DeprecatedConstant( "current", "SensorDeviceClass.CURRENT", "2025.1", ) -_DEPRECATED_DEVICE_CLASS_DATE: Final = ("date", "SensorDeviceClass.DATE", "2025.1") -_DEPRECATED_DEVICE_CLASS_ENERGY: Final = ( +_DEPRECATED_DEVICE_CLASS_DATE: Final = DeprecatedConstant( + "date", "SensorDeviceClass.DATE", "2025.1" +) +_DEPRECATED_DEVICE_CLASS_ENERGY: Final = DeprecatedConstant( "energy", "SensorDeviceClass.ENERGY", "2025.1", ) -_DEPRECATED_DEVICE_CLASS_FREQUENCY: Final = ( +_DEPRECATED_DEVICE_CLASS_FREQUENCY: Final = DeprecatedConstant( "frequency", "SensorDeviceClass.FREQUENCY", "2025.1", ) -_DEPRECATED_DEVICE_CLASS_GAS: Final = ("gas", "SensorDeviceClass.GAS", "2025.1") -_DEPRECATED_DEVICE_CLASS_HUMIDITY: Final = ( +_DEPRECATED_DEVICE_CLASS_GAS: Final = DeprecatedConstant( + "gas", "SensorDeviceClass.GAS", "2025.1" +) +_DEPRECATED_DEVICE_CLASS_HUMIDITY: Final = DeprecatedConstant( "humidity", "SensorDeviceClass.HUMIDITY", "2025.1", ) -_DEPRECATED_DEVICE_CLASS_ILLUMINANCE: Final = ( +_DEPRECATED_DEVICE_CLASS_ILLUMINANCE: Final = DeprecatedConstant( "illuminance", "SensorDeviceClass.ILLUMINANCE", "2025.1", ) -_DEPRECATED_DEVICE_CLASS_MONETARY: Final = ( +_DEPRECATED_DEVICE_CLASS_MONETARY: Final = DeprecatedConstant( "monetary", "SensorDeviceClass.MONETARY", "2025.1", ) -_DEPRECATED_DEVICE_CLASS_NITROGEN_DIOXIDE = ( +_DEPRECATED_DEVICE_CLASS_NITROGEN_DIOXIDE: Final = DeprecatedConstant( "nitrogen_dioxide", "SensorDeviceClass.NITROGEN_DIOXIDE", "2025.1", ) -_DEPRECATED_DEVICE_CLASS_NITROGEN_MONOXIDE = ( +_DEPRECATED_DEVICE_CLASS_NITROGEN_MONOXIDE: Final = DeprecatedConstant( "nitrogen_monoxide", "SensorDeviceClass.NITROGEN_MONOXIDE", "2025.1", ) -_DEPRECATED_DEVICE_CLASS_NITROUS_OXIDE = ( +_DEPRECATED_DEVICE_CLASS_NITROUS_OXIDE: Final = DeprecatedConstant( "nitrous_oxide", "SensorDeviceClass.NITROUS_OXIDE", "2025.1", ) -_DEPRECATED_DEVICE_CLASS_OZONE: Final = ("ozone", "SensorDeviceClass.OZONE", "2025.1") -_DEPRECATED_DEVICE_CLASS_PM1: Final = ("pm1", "SensorDeviceClass.PM1", "2025.1") -_DEPRECATED_DEVICE_CLASS_PM10: Final = ("pm10", "SensorDeviceClass.PM10", "2025.1") -_DEPRECATED_DEVICE_CLASS_PM25: Final = ("pm25", "SensorDeviceClass.PM25", "2025.1") -_DEPRECATED_DEVICE_CLASS_POWER_FACTOR: Final = ( +_DEPRECATED_DEVICE_CLASS_OZONE: Final = DeprecatedConstant( + "ozone", "SensorDeviceClass.OZONE", "2025.1" +) +_DEPRECATED_DEVICE_CLASS_PM1: Final = DeprecatedConstant( + "pm1", "SensorDeviceClass.PM1", "2025.1" +) +_DEPRECATED_DEVICE_CLASS_PM10: Final = DeprecatedConstant( + "pm10", "SensorDeviceClass.PM10", "2025.1" +) +_DEPRECATED_DEVICE_CLASS_PM25: Final = DeprecatedConstant( + "pm25", "SensorDeviceClass.PM25", "2025.1" +) +_DEPRECATED_DEVICE_CLASS_POWER_FACTOR: Final = DeprecatedConstant( "power_factor", "SensorDeviceClass.POWER_FACTOR", "2025.1", ) -_DEPRECATED_DEVICE_CLASS_POWER: Final = ("power", "SensorDeviceClass.POWER", "2025.1") -_DEPRECATED_DEVICE_CLASS_PRESSURE: Final = ( +_DEPRECATED_DEVICE_CLASS_POWER: Final = DeprecatedConstant( + "power", "SensorDeviceClass.POWER", "2025.1" +) +_DEPRECATED_DEVICE_CLASS_PRESSURE: Final = DeprecatedConstant( "pressure", "SensorDeviceClass.PRESSURE", "2025.1", ) -_DEPRECATED_DEVICE_CLASS_SIGNAL_STRENGTH: Final = ( +_DEPRECATED_DEVICE_CLASS_SIGNAL_STRENGTH: Final = DeprecatedConstant( "signal_strength", "SensorDeviceClass.SIGNAL_STRENGTH", "2025.1", ) -_DEPRECATED_DEVICE_CLASS_SULPHUR_DIOXIDE = ( +_DEPRECATED_DEVICE_CLASS_SULPHUR_DIOXIDE: Final = DeprecatedConstant( "sulphur_dioxide", "SensorDeviceClass.SULPHUR_DIOXIDE", "2025.1", ) -_DEPRECATED_DEVICE_CLASS_TEMPERATURE: Final = ( +_DEPRECATED_DEVICE_CLASS_TEMPERATURE: Final = DeprecatedConstant( "temperature", "SensorDeviceClass.TEMPERATURE", "2025.1", ) -_DEPRECATED_DEVICE_CLASS_TIMESTAMP: Final = ( +_DEPRECATED_DEVICE_CLASS_TIMESTAMP: Final = DeprecatedConstant( "timestamp", "SensorDeviceClass.TIMESTAMP", "2025.1", ) -_DEPRECATED_DEVICE_CLASS_VOLATILE_ORGANIC_COMPOUNDS = ( +_DEPRECATED_DEVICE_CLASS_VOLATILE_ORGANIC_COMPOUNDS: Final = DeprecatedConstant( "volatile_organic_compounds", "SensorDeviceClass.VOLATILE_ORGANIC_COMPOUNDS", "2025.1", ) -_DEPRECATED_DEVICE_CLASS_VOLTAGE: Final = ( +_DEPRECATED_DEVICE_CLASS_VOLTAGE: Final = DeprecatedConstant( "voltage", "SensorDeviceClass.VOLTAGE", "2025.1", ) -# Can be removed if no deprecated constant are in this module anymore -def __getattr__(name: str) -> Any: - """Check if the not found name is a deprecated constant. - - If it is, print a deprecation warning and return the value of the constant. - Otherwise raise AttributeError. - """ - module_globals = globals() - if f"_DEPRECATED_{name}" not in module_globals: - raise AttributeError(f"Module {__name__} has no attribute {name!r}") - - # Avoid circular import - from .helpers.deprecation import ( # pylint: disable=import-outside-toplevel - check_if_deprecated_constant, - ) - - return check_if_deprecated_constant(name, module_globals) - - -# Can be removed if no deprecated constant are in this module anymore -def __dir__() -> list[str]: - """Return dir() with deprecated constants.""" - # Copied method from homeassistant.helpers.deprecattion#dir_with_deprecated_constants to avoid import cycle - module_globals = globals() - - return list(module_globals) + [ - name.removeprefix("_DEPRECATED_") - for name in module_globals - if name.startswith("_DEPRECATED_") - ] +# Both can be removed if no deprecated constant are in this module anymore +__getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) +__dir__ = partial(dir_with_deprecated_constants, module_globals=globals()) # #### STATES #### @@ -621,7 +618,7 @@ class UnitOfApparentPower(StrEnum): VOLT_AMPERE = "VA" -_DEPRECATED_POWER_VOLT_AMPERE: Final = ( +_DEPRECATED_POWER_VOLT_AMPERE: Final = DeprecatedConstantEnum( UnitOfApparentPower.VOLT_AMPERE, "2025.1", ) @@ -637,17 +634,17 @@ class UnitOfPower(StrEnum): BTU_PER_HOUR = "BTU/h" -_DEPRECATED_POWER_WATT: Final = ( +_DEPRECATED_POWER_WATT: Final = DeprecatedConstantEnum( UnitOfPower.WATT, "2025.1", ) """Deprecated: please use UnitOfPower.WATT.""" -_DEPRECATED_POWER_KILO_WATT: Final = ( +_DEPRECATED_POWER_KILO_WATT: Final = DeprecatedConstantEnum( UnitOfPower.KILO_WATT, "2025.1", ) """Deprecated: please use UnitOfPower.KILO_WATT.""" -_DEPRECATED_POWER_BTU_PER_HOUR: Final = ( +_DEPRECATED_POWER_BTU_PER_HOUR: Final = DeprecatedConstantEnum( UnitOfPower.BTU_PER_HOUR, "2025.1", ) @@ -668,17 +665,17 @@ class UnitOfEnergy(StrEnum): WATT_HOUR = "Wh" -_DEPRECATED_ENERGY_KILO_WATT_HOUR: Final = ( +_DEPRECATED_ENERGY_KILO_WATT_HOUR: Final = DeprecatedConstantEnum( UnitOfEnergy.KILO_WATT_HOUR, "2025.1", ) """Deprecated: please use UnitOfEnergy.KILO_WATT_HOUR.""" -_DEPRECATED_ENERGY_MEGA_WATT_HOUR: Final = ( +_DEPRECATED_ENERGY_MEGA_WATT_HOUR: Final = DeprecatedConstantEnum( UnitOfEnergy.MEGA_WATT_HOUR, "2025.1", ) """Deprecated: please use UnitOfEnergy.MEGA_WATT_HOUR.""" -_DEPRECATED_ENERGY_WATT_HOUR: Final = ( +_DEPRECATED_ENERGY_WATT_HOUR: Final = DeprecatedConstantEnum( UnitOfEnergy.WATT_HOUR, "2025.1", ) @@ -693,12 +690,12 @@ class UnitOfElectricCurrent(StrEnum): AMPERE = "A" -_DEPRECATED_ELECTRIC_CURRENT_MILLIAMPERE: Final = ( +_DEPRECATED_ELECTRIC_CURRENT_MILLIAMPERE: Final = DeprecatedConstantEnum( UnitOfElectricCurrent.MILLIAMPERE, "2025.1", ) """Deprecated: please use UnitOfElectricCurrent.MILLIAMPERE.""" -_DEPRECATED_ELECTRIC_CURRENT_AMPERE: Final = ( +_DEPRECATED_ELECTRIC_CURRENT_AMPERE: Final = DeprecatedConstantEnum( UnitOfElectricCurrent.AMPERE, "2025.1", ) @@ -713,12 +710,12 @@ class UnitOfElectricPotential(StrEnum): VOLT = "V" -_DEPRECATED_ELECTRIC_POTENTIAL_MILLIVOLT: Final = ( +_DEPRECATED_ELECTRIC_POTENTIAL_MILLIVOLT: Final = DeprecatedConstantEnum( UnitOfElectricPotential.MILLIVOLT, "2025.1", ) """Deprecated: please use UnitOfElectricPotential.MILLIVOLT.""" -_DEPRECATED_ELECTRIC_POTENTIAL_VOLT: Final = ( +_DEPRECATED_ELECTRIC_POTENTIAL_VOLT: Final = DeprecatedConstantEnum( UnitOfElectricPotential.VOLT, "2025.1", ) @@ -742,17 +739,17 @@ class UnitOfTemperature(StrEnum): KELVIN = "K" -_DEPRECATED_TEMP_CELSIUS: Final = ( +_DEPRECATED_TEMP_CELSIUS: Final = DeprecatedConstantEnum( UnitOfTemperature.CELSIUS, "2025.1", ) """Deprecated: please use UnitOfTemperature.CELSIUS""" -_DEPRECATED_TEMP_FAHRENHEIT: Final = ( +_DEPRECATED_TEMP_FAHRENHEIT: Final = DeprecatedConstantEnum( UnitOfTemperature.FAHRENHEIT, "2025.1", ) """Deprecated: please use UnitOfTemperature.FAHRENHEIT""" -_DEPRECATED_TEMP_KELVIN: Final = ( +_DEPRECATED_TEMP_KELVIN: Final = DeprecatedConstantEnum( UnitOfTemperature.KELVIN, "2025.1", ) @@ -774,47 +771,47 @@ class UnitOfTime(StrEnum): YEARS = "y" -_DEPRECATED_TIME_MICROSECONDS: Final = ( +_DEPRECATED_TIME_MICROSECONDS: Final = DeprecatedConstantEnum( UnitOfTime.MICROSECONDS, "2025.1", ) """Deprecated: please use UnitOfTime.MICROSECONDS.""" -_DEPRECATED_TIME_MILLISECONDS: Final = ( +_DEPRECATED_TIME_MILLISECONDS: Final = DeprecatedConstantEnum( UnitOfTime.MILLISECONDS, "2025.1", ) """Deprecated: please use UnitOfTime.MILLISECONDS.""" -_DEPRECATED_TIME_SECONDS: Final = ( +_DEPRECATED_TIME_SECONDS: Final = DeprecatedConstantEnum( UnitOfTime.SECONDS, "2025.1", ) """Deprecated: please use UnitOfTime.SECONDS.""" -_DEPRECATED_TIME_MINUTES: Final = ( +_DEPRECATED_TIME_MINUTES: Final = DeprecatedConstantEnum( UnitOfTime.MINUTES, "2025.1", ) """Deprecated: please use UnitOfTime.MINUTES.""" -_DEPRECATED_TIME_HOURS: Final = ( +_DEPRECATED_TIME_HOURS: Final = DeprecatedConstantEnum( UnitOfTime.HOURS, "2025.1", ) """Deprecated: please use UnitOfTime.HOURS.""" -_DEPRECATED_TIME_DAYS: Final = ( +_DEPRECATED_TIME_DAYS: Final = DeprecatedConstantEnum( UnitOfTime.DAYS, "2025.1", ) """Deprecated: please use UnitOfTime.DAYS.""" -_DEPRECATED_TIME_WEEKS: Final = ( +_DEPRECATED_TIME_WEEKS: Final = DeprecatedConstantEnum( UnitOfTime.WEEKS, "2025.1", ) """Deprecated: please use UnitOfTime.WEEKS.""" -_DEPRECATED_TIME_MONTHS: Final = ( +_DEPRECATED_TIME_MONTHS: Final = DeprecatedConstantEnum( UnitOfTime.MONTHS, "2025.1", ) """Deprecated: please use UnitOfTime.MONTHS.""" -_DEPRECATED_TIME_YEARS: Final = ( +_DEPRECATED_TIME_YEARS: Final = DeprecatedConstantEnum( UnitOfTime.YEARS, "2025.1", ) @@ -835,42 +832,42 @@ class UnitOfLength(StrEnum): MILES = "mi" -_DEPRECATED_LENGTH_MILLIMETERS: Final = ( +_DEPRECATED_LENGTH_MILLIMETERS: Final = DeprecatedConstantEnum( UnitOfLength.MILLIMETERS, "2025.1", ) """Deprecated: please use UnitOfLength.MILLIMETERS.""" -_DEPRECATED_LENGTH_CENTIMETERS: Final = ( +_DEPRECATED_LENGTH_CENTIMETERS: Final = DeprecatedConstantEnum( UnitOfLength.CENTIMETERS, "2025.1", ) """Deprecated: please use UnitOfLength.CENTIMETERS.""" -_DEPRECATED_LENGTH_METERS: Final = ( +_DEPRECATED_LENGTH_METERS: Final = DeprecatedConstantEnum( UnitOfLength.METERS, "2025.1", ) """Deprecated: please use UnitOfLength.METERS.""" -_DEPRECATED_LENGTH_KILOMETERS: Final = ( +_DEPRECATED_LENGTH_KILOMETERS: Final = DeprecatedConstantEnum( UnitOfLength.KILOMETERS, "2025.1", ) """Deprecated: please use UnitOfLength.KILOMETERS.""" -_DEPRECATED_LENGTH_INCHES: Final = ( +_DEPRECATED_LENGTH_INCHES: Final = DeprecatedConstantEnum( UnitOfLength.INCHES, "2025.1", ) """Deprecated: please use UnitOfLength.INCHES.""" -_DEPRECATED_LENGTH_FEET: Final = ( +_DEPRECATED_LENGTH_FEET: Final = DeprecatedConstantEnum( UnitOfLength.FEET, "2025.1", ) """Deprecated: please use UnitOfLength.FEET.""" -_DEPRECATED_LENGTH_YARD: Final = ( +_DEPRECATED_LENGTH_YARD: Final = DeprecatedConstantEnum( UnitOfLength.YARDS, "2025.1", ) """Deprecated: please use UnitOfLength.YARDS.""" -_DEPRECATED_LENGTH_MILES: Final = ( +_DEPRECATED_LENGTH_MILES: Final = DeprecatedConstantEnum( UnitOfLength.MILES, "2025.1", ) @@ -887,22 +884,22 @@ class UnitOfFrequency(StrEnum): GIGAHERTZ = "GHz" -_DEPRECATED_FREQUENCY_HERTZ: Final = ( +_DEPRECATED_FREQUENCY_HERTZ: Final = DeprecatedConstantEnum( UnitOfFrequency.HERTZ, "2025.1", ) """Deprecated: please use UnitOfFrequency.HERTZ""" -_DEPRECATED_FREQUENCY_KILOHERTZ: Final = ( +_DEPRECATED_FREQUENCY_KILOHERTZ: Final = DeprecatedConstantEnum( UnitOfFrequency.KILOHERTZ, "2025.1", ) """Deprecated: please use UnitOfFrequency.KILOHERTZ""" -_DEPRECATED_FREQUENCY_MEGAHERTZ: Final = ( +_DEPRECATED_FREQUENCY_MEGAHERTZ: Final = DeprecatedConstantEnum( UnitOfFrequency.MEGAHERTZ, "2025.1", ) """Deprecated: please use UnitOfFrequency.MEGAHERTZ""" -_DEPRECATED_FREQUENCY_GIGAHERTZ: Final = ( +_DEPRECATED_FREQUENCY_GIGAHERTZ: Final = DeprecatedConstantEnum( UnitOfFrequency.GIGAHERTZ, "2025.1", ) @@ -924,47 +921,47 @@ class UnitOfPressure(StrEnum): PSI = "psi" -_DEPRECATED_PRESSURE_PA: Final = ( +_DEPRECATED_PRESSURE_PA: Final = DeprecatedConstantEnum( UnitOfPressure.PA, "2025.1", ) """Deprecated: please use UnitOfPressure.PA""" -_DEPRECATED_PRESSURE_HPA: Final = ( +_DEPRECATED_PRESSURE_HPA: Final = DeprecatedConstantEnum( UnitOfPressure.HPA, "2025.1", ) """Deprecated: please use UnitOfPressure.HPA""" -_DEPRECATED_PRESSURE_KPA: Final = ( +_DEPRECATED_PRESSURE_KPA: Final = DeprecatedConstantEnum( UnitOfPressure.KPA, "2025.1", ) """Deprecated: please use UnitOfPressure.KPA""" -_DEPRECATED_PRESSURE_BAR: Final = ( +_DEPRECATED_PRESSURE_BAR: Final = DeprecatedConstantEnum( UnitOfPressure.BAR, "2025.1", ) """Deprecated: please use UnitOfPressure.BAR""" -_DEPRECATED_PRESSURE_CBAR: Final = ( +_DEPRECATED_PRESSURE_CBAR: Final = DeprecatedConstantEnum( UnitOfPressure.CBAR, "2025.1", ) """Deprecated: please use UnitOfPressure.CBAR""" -_DEPRECATED_PRESSURE_MBAR: Final = ( +_DEPRECATED_PRESSURE_MBAR: Final = DeprecatedConstantEnum( UnitOfPressure.MBAR, "2025.1", ) """Deprecated: please use UnitOfPressure.MBAR""" -_DEPRECATED_PRESSURE_MMHG: Final = ( +_DEPRECATED_PRESSURE_MMHG: Final = DeprecatedConstantEnum( UnitOfPressure.MMHG, "2025.1", ) """Deprecated: please use UnitOfPressure.MMHG""" -_DEPRECATED_PRESSURE_INHG: Final = ( +_DEPRECATED_PRESSURE_INHG: Final = DeprecatedConstantEnum( UnitOfPressure.INHG, "2025.1", ) """Deprecated: please use UnitOfPressure.INHG""" -_DEPRECATED_PRESSURE_PSI: Final = ( +_DEPRECATED_PRESSURE_PSI: Final = DeprecatedConstantEnum( UnitOfPressure.PSI, "2025.1", ) @@ -979,12 +976,12 @@ class UnitOfSoundPressure(StrEnum): WEIGHTED_DECIBEL_A = "dBA" -_DEPRECATED_SOUND_PRESSURE_DB: Final = ( +_DEPRECATED_SOUND_PRESSURE_DB: Final = DeprecatedConstantEnum( UnitOfSoundPressure.DECIBEL, "2025.1", ) """Deprecated: please use UnitOfSoundPressure.DECIBEL""" -_DEPRECATED_SOUND_PRESSURE_WEIGHTED_DBA: Final = ( +_DEPRECATED_SOUND_PRESSURE_WEIGHTED_DBA: Final = DeprecatedConstantEnum( UnitOfSoundPressure.WEIGHTED_DECIBEL_A, "2025.1", ) @@ -1010,33 +1007,33 @@ class UnitOfVolume(StrEnum): British/Imperial fluid ounces are not yet supported""" -_DEPRECATED_VOLUME_LITERS: Final = ( +_DEPRECATED_VOLUME_LITERS: Final = DeprecatedConstantEnum( UnitOfVolume.LITERS, "2025.1", ) """Deprecated: please use UnitOfVolume.LITERS""" -_DEPRECATED_VOLUME_MILLILITERS: Final = ( +_DEPRECATED_VOLUME_MILLILITERS: Final = DeprecatedConstantEnum( UnitOfVolume.MILLILITERS, "2025.1", ) """Deprecated: please use UnitOfVolume.MILLILITERS""" -_DEPRECATED_VOLUME_CUBIC_METERS: Final = ( +_DEPRECATED_VOLUME_CUBIC_METERS: Final = DeprecatedConstantEnum( UnitOfVolume.CUBIC_METERS, "2025.1", ) """Deprecated: please use UnitOfVolume.CUBIC_METERS""" -_DEPRECATED_VOLUME_CUBIC_FEET: Final = ( +_DEPRECATED_VOLUME_CUBIC_FEET: Final = DeprecatedConstantEnum( UnitOfVolume.CUBIC_FEET, "2025.1", ) """Deprecated: please use UnitOfVolume.CUBIC_FEET""" -_DEPRECATED_VOLUME_GALLONS: Final = ( +_DEPRECATED_VOLUME_GALLONS: Final = DeprecatedConstantEnum( UnitOfVolume.GALLONS, "2025.1", ) """Deprecated: please use UnitOfVolume.GALLONS""" -_DEPRECATED_VOLUME_FLUID_OUNCE: Final = ( +_DEPRECATED_VOLUME_FLUID_OUNCE: Final = DeprecatedConstantEnum( UnitOfVolume.FLUID_OUNCES, "2025.1", ) @@ -1051,12 +1048,12 @@ class UnitOfVolumeFlowRate(StrEnum): CUBIC_FEET_PER_MINUTE = "ft³/m" -_DEPRECATED_VOLUME_FLOW_RATE_CUBIC_METERS_PER_HOUR: Final = ( +_DEPRECATED_VOLUME_FLOW_RATE_CUBIC_METERS_PER_HOUR: Final = DeprecatedConstantEnum( UnitOfVolumeFlowRate.CUBIC_METERS_PER_HOUR, "2025.1", ) """Deprecated: please use UnitOfVolumeFlowRate.CUBIC_METERS_PER_HOUR""" -_DEPRECATED_VOLUME_FLOW_RATE_CUBIC_FEET_PER_MINUTE: Final = ( +_DEPRECATED_VOLUME_FLOW_RATE_CUBIC_FEET_PER_MINUTE: Final = DeprecatedConstantEnum( UnitOfVolumeFlowRate.CUBIC_FEET_PER_MINUTE, "2025.1", ) @@ -1079,32 +1076,32 @@ class UnitOfMass(StrEnum): STONES = "st" -_DEPRECATED_MASS_GRAMS: Final = ( +_DEPRECATED_MASS_GRAMS: Final = DeprecatedConstantEnum( UnitOfMass.GRAMS, "2025.1", ) """Deprecated: please use UnitOfMass.GRAMS""" -_DEPRECATED_MASS_KILOGRAMS: Final = ( +_DEPRECATED_MASS_KILOGRAMS: Final = DeprecatedConstantEnum( UnitOfMass.KILOGRAMS, "2025.1", ) """Deprecated: please use UnitOfMass.KILOGRAMS""" -_DEPRECATED_MASS_MILLIGRAMS: Final = ( +_DEPRECATED_MASS_MILLIGRAMS: Final = DeprecatedConstantEnum( UnitOfMass.MILLIGRAMS, "2025.1", ) """Deprecated: please use UnitOfMass.MILLIGRAMS""" -_DEPRECATED_MASS_MICROGRAMS: Final = ( +_DEPRECATED_MASS_MICROGRAMS: Final = DeprecatedConstantEnum( UnitOfMass.MICROGRAMS, "2025.1", ) """Deprecated: please use UnitOfMass.MICROGRAMS""" -_DEPRECATED_MASS_OUNCES: Final = ( +_DEPRECATED_MASS_OUNCES: Final = DeprecatedConstantEnum( UnitOfMass.OUNCES, "2025.1", ) """Deprecated: please use UnitOfMass.OUNCES""" -_DEPRECATED_MASS_POUNDS: Final = ( +_DEPRECATED_MASS_POUNDS: Final = DeprecatedConstantEnum( UnitOfMass.POUNDS, "2025.1", ) @@ -1135,12 +1132,12 @@ class UnitOfIrradiance(StrEnum): # Irradiation units -_DEPRECATED_IRRADIATION_WATTS_PER_SQUARE_METER: Final = ( +_DEPRECATED_IRRADIATION_WATTS_PER_SQUARE_METER: Final = DeprecatedConstantEnum( UnitOfIrradiance.WATTS_PER_SQUARE_METER, "2025.1", ) """Deprecated: please use UnitOfIrradiance.WATTS_PER_SQUARE_METER""" -_DEPRECATED_IRRADIATION_BTUS_PER_HOUR_SQUARE_FOOT: Final = ( +_DEPRECATED_IRRADIATION_BTUS_PER_HOUR_SQUARE_FOOT: Final = DeprecatedConstantEnum( UnitOfIrradiance.BTUS_PER_HOUR_SQUARE_FOOT, "2025.1", ) @@ -1185,19 +1182,21 @@ class UnitOfPrecipitationDepth(StrEnum): # Precipitation units -_DEPRECATED_PRECIPITATION_INCHES: Final = (UnitOfPrecipitationDepth.INCHES, "2025.1") +_DEPRECATED_PRECIPITATION_INCHES: Final = DeprecatedConstantEnum( + UnitOfPrecipitationDepth.INCHES, "2025.1" +) """Deprecated: please use UnitOfPrecipitationDepth.INCHES""" -_DEPRECATED_PRECIPITATION_MILLIMETERS: Final = ( +_DEPRECATED_PRECIPITATION_MILLIMETERS: Final = DeprecatedConstantEnum( UnitOfPrecipitationDepth.MILLIMETERS, "2025.1", ) """Deprecated: please use UnitOfPrecipitationDepth.MILLIMETERS""" -_DEPRECATED_PRECIPITATION_MILLIMETERS_PER_HOUR: Final = ( +_DEPRECATED_PRECIPITATION_MILLIMETERS_PER_HOUR: Final = DeprecatedConstantEnum( UnitOfVolumetricFlux.MILLIMETERS_PER_HOUR, "2025.1", ) """Deprecated: please use UnitOfVolumetricFlux.MILLIMETERS_PER_HOUR""" -_DEPRECATED_PRECIPITATION_INCHES_PER_HOUR: Final = ( +_DEPRECATED_PRECIPITATION_INCHES_PER_HOUR: Final = DeprecatedConstantEnum( UnitOfVolumetricFlux.INCHES_PER_HOUR, "2025.1", ) @@ -1223,33 +1222,39 @@ class UnitOfSpeed(StrEnum): MILES_PER_HOUR = "mph" -_DEPRECATED_SPEED_FEET_PER_SECOND: Final = (UnitOfSpeed.FEET_PER_SECOND, "2025.1") +_DEPRECATED_SPEED_FEET_PER_SECOND: Final = DeprecatedConstantEnum( + UnitOfSpeed.FEET_PER_SECOND, "2025.1" +) """Deprecated: please use UnitOfSpeed.FEET_PER_SECOND""" -_DEPRECATED_SPEED_METERS_PER_SECOND: Final = (UnitOfSpeed.METERS_PER_SECOND, "2025.1") +_DEPRECATED_SPEED_METERS_PER_SECOND: Final = DeprecatedConstantEnum( + UnitOfSpeed.METERS_PER_SECOND, "2025.1" +) """Deprecated: please use UnitOfSpeed.METERS_PER_SECOND""" -_DEPRECATED_SPEED_KILOMETERS_PER_HOUR: Final = ( +_DEPRECATED_SPEED_KILOMETERS_PER_HOUR: Final = DeprecatedConstantEnum( UnitOfSpeed.KILOMETERS_PER_HOUR, "2025.1", ) """Deprecated: please use UnitOfSpeed.KILOMETERS_PER_HOUR""" -_DEPRECATED_SPEED_KNOTS: Final = (UnitOfSpeed.KNOTS, "2025.1") +_DEPRECATED_SPEED_KNOTS: Final = DeprecatedConstantEnum(UnitOfSpeed.KNOTS, "2025.1") """Deprecated: please use UnitOfSpeed.KNOTS""" -_DEPRECATED_SPEED_MILES_PER_HOUR: Final = (UnitOfSpeed.MILES_PER_HOUR, "2025.1") +_DEPRECATED_SPEED_MILES_PER_HOUR: Final = DeprecatedConstantEnum( + UnitOfSpeed.MILES_PER_HOUR, "2025.1" +) """Deprecated: please use UnitOfSpeed.MILES_PER_HOUR""" -_DEPRECATED_SPEED_MILLIMETERS_PER_DAY: Final = ( +_DEPRECATED_SPEED_MILLIMETERS_PER_DAY: Final = DeprecatedConstantEnum( UnitOfVolumetricFlux.MILLIMETERS_PER_DAY, "2025.1", ) """Deprecated: please use UnitOfVolumetricFlux.MILLIMETERS_PER_DAY""" -_DEPRECATED_SPEED_INCHES_PER_DAY: Final = ( +_DEPRECATED_SPEED_INCHES_PER_DAY: Final = DeprecatedConstantEnum( UnitOfVolumetricFlux.INCHES_PER_DAY, "2025.1", ) """Deprecated: please use UnitOfVolumetricFlux.INCHES_PER_DAY""" -_DEPRECATED_SPEED_INCHES_PER_HOUR: Final = ( +_DEPRECATED_SPEED_INCHES_PER_HOUR: Final = DeprecatedConstantEnum( UnitOfVolumetricFlux.INCHES_PER_HOUR, "2025.1", ) @@ -1288,47 +1293,87 @@ class UnitOfInformation(StrEnum): YOBIBYTES = "YiB" -_DEPRECATED_DATA_BITS: Final = (UnitOfInformation.BITS, "2025.1") +_DEPRECATED_DATA_BITS: Final = DeprecatedConstantEnum(UnitOfInformation.BITS, "2025.1") """Deprecated: please use UnitOfInformation.BITS""" -_DEPRECATED_DATA_KILOBITS: Final = (UnitOfInformation.KILOBITS, "2025.1") +_DEPRECATED_DATA_KILOBITS: Final = DeprecatedConstantEnum( + UnitOfInformation.KILOBITS, "2025.1" +) """Deprecated: please use UnitOfInformation.KILOBITS""" -_DEPRECATED_DATA_MEGABITS: Final = (UnitOfInformation.MEGABITS, "2025.1") +_DEPRECATED_DATA_MEGABITS: Final = DeprecatedConstantEnum( + UnitOfInformation.MEGABITS, "2025.1" +) """Deprecated: please use UnitOfInformation.MEGABITS""" -_DEPRECATED_DATA_GIGABITS: Final = (UnitOfInformation.GIGABITS, "2025.1") +_DEPRECATED_DATA_GIGABITS: Final = DeprecatedConstantEnum( + UnitOfInformation.GIGABITS, "2025.1" +) """Deprecated: please use UnitOfInformation.GIGABITS""" -_DEPRECATED_DATA_BYTES: Final = (UnitOfInformation.BYTES, "2025.1") +_DEPRECATED_DATA_BYTES: Final = DeprecatedConstantEnum( + UnitOfInformation.BYTES, "2025.1" +) """Deprecated: please use UnitOfInformation.BYTES""" -_DEPRECATED_DATA_KILOBYTES: Final = (UnitOfInformation.KILOBYTES, "2025.1") +_DEPRECATED_DATA_KILOBYTES: Final = DeprecatedConstantEnum( + UnitOfInformation.KILOBYTES, "2025.1" +) """Deprecated: please use UnitOfInformation.KILOBYTES""" -_DEPRECATED_DATA_MEGABYTES: Final = (UnitOfInformation.MEGABYTES, "2025.1") +_DEPRECATED_DATA_MEGABYTES: Final = DeprecatedConstantEnum( + UnitOfInformation.MEGABYTES, "2025.1" +) """Deprecated: please use UnitOfInformation.MEGABYTES""" -_DEPRECATED_DATA_GIGABYTES: Final = (UnitOfInformation.GIGABYTES, "2025.1") +_DEPRECATED_DATA_GIGABYTES: Final = DeprecatedConstantEnum( + UnitOfInformation.GIGABYTES, "2025.1" +) """Deprecated: please use UnitOfInformation.GIGABYTES""" -_DEPRECATED_DATA_TERABYTES: Final = (UnitOfInformation.TERABYTES, "2025.1") +_DEPRECATED_DATA_TERABYTES: Final = DeprecatedConstantEnum( + UnitOfInformation.TERABYTES, "2025.1" +) """Deprecated: please use UnitOfInformation.TERABYTES""" -_DEPRECATED_DATA_PETABYTES: Final = (UnitOfInformation.PETABYTES, "2025.1") +_DEPRECATED_DATA_PETABYTES: Final = DeprecatedConstantEnum( + UnitOfInformation.PETABYTES, "2025.1" +) """Deprecated: please use UnitOfInformation.PETABYTES""" -_DEPRECATED_DATA_EXABYTES: Final = (UnitOfInformation.EXABYTES, "2025.1") +_DEPRECATED_DATA_EXABYTES: Final = DeprecatedConstantEnum( + UnitOfInformation.EXABYTES, "2025.1" +) """Deprecated: please use UnitOfInformation.EXABYTES""" -_DEPRECATED_DATA_ZETTABYTES: Final = (UnitOfInformation.ZETTABYTES, "2025.1") +_DEPRECATED_DATA_ZETTABYTES: Final = DeprecatedConstantEnum( + UnitOfInformation.ZETTABYTES, "2025.1" +) """Deprecated: please use UnitOfInformation.ZETTABYTES""" -_DEPRECATED_DATA_YOTTABYTES: Final = (UnitOfInformation.YOTTABYTES, "2025.1") +_DEPRECATED_DATA_YOTTABYTES: Final = DeprecatedConstantEnum( + UnitOfInformation.YOTTABYTES, "2025.1" +) """Deprecated: please use UnitOfInformation.YOTTABYTES""" -_DEPRECATED_DATA_KIBIBYTES: Final = (UnitOfInformation.KIBIBYTES, "2025.1") +_DEPRECATED_DATA_KIBIBYTES: Final = DeprecatedConstantEnum( + UnitOfInformation.KIBIBYTES, "2025.1" +) """Deprecated: please use UnitOfInformation.KIBIBYTES""" -_DEPRECATED_DATA_MEBIBYTES: Final = (UnitOfInformation.MEBIBYTES, "2025.1") +_DEPRECATED_DATA_MEBIBYTES: Final = DeprecatedConstantEnum( + UnitOfInformation.MEBIBYTES, "2025.1" +) """Deprecated: please use UnitOfInformation.MEBIBYTES""" -_DEPRECATED_DATA_GIBIBYTES: Final = (UnitOfInformation.GIBIBYTES, "2025.1") +_DEPRECATED_DATA_GIBIBYTES: Final = DeprecatedConstantEnum( + UnitOfInformation.GIBIBYTES, "2025.1" +) """Deprecated: please use UnitOfInformation.GIBIBYTES""" -_DEPRECATED_DATA_TEBIBYTES: Final = (UnitOfInformation.TEBIBYTES, "2025.1") +_DEPRECATED_DATA_TEBIBYTES: Final = DeprecatedConstantEnum( + UnitOfInformation.TEBIBYTES, "2025.1" +) """Deprecated: please use UnitOfInformation.TEBIBYTES""" -_DEPRECATED_DATA_PEBIBYTES: Final = (UnitOfInformation.PEBIBYTES, "2025.1") +_DEPRECATED_DATA_PEBIBYTES: Final = DeprecatedConstantEnum( + UnitOfInformation.PEBIBYTES, "2025.1" +) """Deprecated: please use UnitOfInformation.PEBIBYTES""" -_DEPRECATED_DATA_EXBIBYTES: Final = (UnitOfInformation.EXBIBYTES, "2025.1") +_DEPRECATED_DATA_EXBIBYTES: Final = DeprecatedConstantEnum( + UnitOfInformation.EXBIBYTES, "2025.1" +) """Deprecated: please use UnitOfInformation.EXBIBYTES""" -_DEPRECATED_DATA_ZEBIBYTES: Final = (UnitOfInformation.ZEBIBYTES, "2025.1") +_DEPRECATED_DATA_ZEBIBYTES: Final = DeprecatedConstantEnum( + UnitOfInformation.ZEBIBYTES, "2025.1" +) """Deprecated: please use UnitOfInformation.ZEBIBYTES""" -_DEPRECATED_DATA_YOBIBYTES: Final = (UnitOfInformation.YOBIBYTES, "2025.1") +_DEPRECATED_DATA_YOBIBYTES: Final = DeprecatedConstantEnum( + UnitOfInformation.YOBIBYTES, "2025.1" +) """Deprecated: please use UnitOfInformation.YOBIBYTES""" @@ -1349,57 +1394,57 @@ class UnitOfDataRate(StrEnum): GIBIBYTES_PER_SECOND = "GiB/s" -_DEPRECATED_DATA_RATE_BITS_PER_SECOND: Final = ( +_DEPRECATED_DATA_RATE_BITS_PER_SECOND: Final = DeprecatedConstantEnum( UnitOfDataRate.BITS_PER_SECOND, "2025.1", ) """Deprecated: please use UnitOfDataRate.BITS_PER_SECOND""" -_DEPRECATED_DATA_RATE_KILOBITS_PER_SECOND: Final = ( +_DEPRECATED_DATA_RATE_KILOBITS_PER_SECOND: Final = DeprecatedConstantEnum( UnitOfDataRate.KILOBITS_PER_SECOND, "2025.1", ) """Deprecated: please use UnitOfDataRate.KILOBITS_PER_SECOND""" -_DEPRECATED_DATA_RATE_MEGABITS_PER_SECOND: Final = ( +_DEPRECATED_DATA_RATE_MEGABITS_PER_SECOND: Final = DeprecatedConstantEnum( UnitOfDataRate.MEGABITS_PER_SECOND, "2025.1", ) """Deprecated: please use UnitOfDataRate.MEGABITS_PER_SECOND""" -_DEPRECATED_DATA_RATE_GIGABITS_PER_SECOND: Final = ( +_DEPRECATED_DATA_RATE_GIGABITS_PER_SECOND: Final = DeprecatedConstantEnum( UnitOfDataRate.GIGABITS_PER_SECOND, "2025.1", ) """Deprecated: please use UnitOfDataRate.GIGABITS_PER_SECOND""" -_DEPRECATED_DATA_RATE_BYTES_PER_SECOND: Final = ( +_DEPRECATED_DATA_RATE_BYTES_PER_SECOND: Final = DeprecatedConstantEnum( UnitOfDataRate.BYTES_PER_SECOND, "2025.1", ) """Deprecated: please use UnitOfDataRate.BYTES_PER_SECOND""" -_DEPRECATED_DATA_RATE_KILOBYTES_PER_SECOND: Final = ( +_DEPRECATED_DATA_RATE_KILOBYTES_PER_SECOND: Final = DeprecatedConstantEnum( UnitOfDataRate.KILOBYTES_PER_SECOND, "2025.1", ) """Deprecated: please use UnitOfDataRate.KILOBYTES_PER_SECOND""" -_DEPRECATED_DATA_RATE_MEGABYTES_PER_SECOND: Final = ( +_DEPRECATED_DATA_RATE_MEGABYTES_PER_SECOND: Final = DeprecatedConstantEnum( UnitOfDataRate.MEGABYTES_PER_SECOND, "2025.1", ) """Deprecated: please use UnitOfDataRate.MEGABYTES_PER_SECOND""" -_DEPRECATED_DATA_RATE_GIGABYTES_PER_SECOND: Final = ( +_DEPRECATED_DATA_RATE_GIGABYTES_PER_SECOND: Final = DeprecatedConstantEnum( UnitOfDataRate.GIGABYTES_PER_SECOND, "2025.1", ) """Deprecated: please use UnitOfDataRate.GIGABYTES_PER_SECOND""" -_DEPRECATED_DATA_RATE_KIBIBYTES_PER_SECOND: Final = ( +_DEPRECATED_DATA_RATE_KIBIBYTES_PER_SECOND: Final = DeprecatedConstantEnum( UnitOfDataRate.KIBIBYTES_PER_SECOND, "2025.1", ) """Deprecated: please use UnitOfDataRate.KIBIBYTES_PER_SECOND""" -_DEPRECATED_DATA_RATE_MEBIBYTES_PER_SECOND: Final = ( +_DEPRECATED_DATA_RATE_MEBIBYTES_PER_SECOND: Final = DeprecatedConstantEnum( UnitOfDataRate.MEBIBYTES_PER_SECOND, "2025.1", ) """Deprecated: please use UnitOfDataRate.MEBIBYTES_PER_SECOND""" -_DEPRECATED_DATA_RATE_GIBIBYTES_PER_SECOND: Final = ( +_DEPRECATED_DATA_RATE_GIBIBYTES_PER_SECOND: Final = DeprecatedConstantEnum( UnitOfDataRate.GIBIBYTES_PER_SECOND, "2025.1", ) @@ -1540,8 +1585,12 @@ class EntityCategory(StrEnum): # ENTITY_CATEGOR* below are deprecated as of 2021.12 # use the EntityCategory enum instead. -_DEPRECATED_ENTITY_CATEGORY_CONFIG: Final = (EntityCategory.CONFIG, "2025.1") -_DEPRECATED_ENTITY_CATEGORY_DIAGNOSTIC: Final = (EntityCategory.DIAGNOSTIC, "2025.1") +_DEPRECATED_ENTITY_CATEGORY_CONFIG: Final = DeprecatedConstantEnum( + EntityCategory.CONFIG, "2025.1" +) +_DEPRECATED_ENTITY_CATEGORY_DIAGNOSTIC: Final = DeprecatedConstantEnum( + EntityCategory.DIAGNOSTIC, "2025.1" +) ENTITY_CATEGORIES: Final[list[str]] = [cls.value for cls in EntityCategory] # The ID of the Home Assistant Media Player Cast App diff --git a/homeassistant/core.py b/homeassistant/core.py index 51cb3d4e496d53..b15d393c63e237 100644 --- a/homeassistant/core.py +++ b/homeassistant/core.py @@ -81,6 +81,11 @@ ServiceNotFound, Unauthorized, ) +from .helpers.deprecation import ( + DeprecatedConstantEnum, + check_if_deprecated_constant, + dir_with_deprecated_constants, +) from .helpers.json import json_dumps from .util import dt as dt_util, location from .util.async_ import ( @@ -147,41 +152,16 @@ class ConfigSource(enum.StrEnum): # SOURCE_* are deprecated as of Home Assistant 2022.2, use ConfigSource instead -_DEPRECATED_SOURCE_DISCOVERED = (ConfigSource.DISCOVERED, "2025.1") -_DEPRECATED_SOURCE_STORAGE = (ConfigSource.STORAGE, "2025.1") -_DEPRECATED_SOURCE_YAML = (ConfigSource.YAML, "2025.1") - - -# Can be removed if no deprecated constant are in this module anymore -def __getattr__(name: str) -> Any: - """Check if the not found name is a deprecated constant. - - If it is, print a deprecation warning and return the value of the constant. - Otherwise raise AttributeError. - """ - module_globals = globals() - if f"_DEPRECATED_{name}" not in module_globals: - raise AttributeError(f"Module {__name__} has no attribute {name!r}") - - # Avoid circular import - from .helpers.deprecation import ( # pylint: disable=import-outside-toplevel - check_if_deprecated_constant, - ) - - return check_if_deprecated_constant(name, module_globals) - +_DEPRECATED_SOURCE_DISCOVERED = DeprecatedConstantEnum( + ConfigSource.DISCOVERED, "2025.1" +) +_DEPRECATED_SOURCE_STORAGE = DeprecatedConstantEnum(ConfigSource.STORAGE, "2025.1") +_DEPRECATED_SOURCE_YAML = DeprecatedConstantEnum(ConfigSource.YAML, "2025.1") -# Can be removed if no deprecated constant are in this module anymore -def __dir__() -> list[str]: - """Return dir() with deprecated constants.""" - # Copied method from homeassistant.helpers.deprecattion#dir_with_deprecated_constants to avoid import cycle - module_globals = globals() - return list(module_globals) + [ - name.removeprefix("_DEPRECATED_") - for name in module_globals - if name.startswith("_DEPRECATED_") - ] +# Both can be removed if no deprecated constant are in this module anymore +__getattr__ = functools.partial(check_if_deprecated_constant, module_globals=globals()) +__dir__ = functools.partial(dir_with_deprecated_constants, module_globals=globals()) # How long to wait until things that run on startup have to finish. diff --git a/homeassistant/helpers/deprecation.py b/homeassistant/helpers/deprecation.py index efd0363732a80f..72b26e90b84275 100644 --- a/homeassistant/helpers/deprecation.py +++ b/homeassistant/helpers/deprecation.py @@ -9,12 +9,6 @@ import logging from typing import Any, NamedTuple, ParamSpec, TypeVar -from homeassistant.core import HomeAssistant, async_get_hass -from homeassistant.exceptions import HomeAssistantError -from homeassistant.loader import async_suggest_report_issue - -from .frame import MissingIntegrationFrame, get_integration_frame - _ObjectT = TypeVar("_ObjectT", bound=object) _R = TypeVar("_R") _P = ParamSpec("_P") @@ -175,6 +169,13 @@ def _print_deprecation_warning_internal( *, log_when_no_integration_is_found: bool, ) -> None: + # pylint: disable=import-outside-toplevel + from homeassistant.core import HomeAssistant, async_get_hass + from homeassistant.exceptions import HomeAssistantError + from homeassistant.loader import async_suggest_report_issue + + from .frame import MissingIntegrationFrame, get_integration_frame + logger = logging.getLogger(module_name) if breaks_in_ha_version: breaks_in = f" which will be removed in HA Core {breaks_in_ha_version}" @@ -265,18 +266,6 @@ def check_if_deprecated_constant(name: str, module_globals: dict[str, Any]) -> A f"{deprecated_const.enum.__class__.__name__}.{deprecated_const.enum.name}" ) breaks_in_ha_version = deprecated_const.breaks_in_ha_version - elif isinstance(deprecated_const, tuple): - # Use DeprecatedConstant and DeprecatedConstant instead, where possible - # Used to avoid import cycles. - if len(deprecated_const) == 3: - value = deprecated_const[0] - replacement = deprecated_const[1] - breaks_in_ha_version = deprecated_const[2] - elif len(deprecated_const) == 2 and isinstance(deprecated_const[0], Enum): - enum = deprecated_const[0] - value = enum.value - replacement = f"{enum.__class__.__name__}.{enum.name}" - breaks_in_ha_version = deprecated_const[1] if value is None or replacement is None: msg = ( diff --git a/tests/helpers/test_deprecation.py b/tests/helpers/test_deprecation.py index bd3546afb124a4..017e541bb08e4f 100644 --- a/tests/helpers/test_deprecation.py +++ b/tests/helpers/test_deprecation.py @@ -299,22 +299,6 @@ def _get_value(obj: DeprecatedConstant | DeprecatedConstantEnum | tuple) -> Any: DeprecatedConstantEnum(TestDeprecatedConstantEnum.TEST, "2099.1"), " which will be removed in HA Core 2099.1. Use TestDeprecatedConstantEnum.TEST instead", ), - ( - ("value", "NEW_CONSTANT", None), - ". Use NEW_CONSTANT instead", - ), - ( - (1, "NEW_CONSTANT", "2099.1"), - " which will be removed in HA Core 2099.1. Use NEW_CONSTANT instead", - ), - ( - (TestDeprecatedConstantEnum.TEST, None), - ". Use TestDeprecatedConstantEnum.TEST instead", - ), - ( - (TestDeprecatedConstantEnum.TEST, "2099.1"), - " which will be removed in HA Core 2099.1. Use TestDeprecatedConstantEnum.TEST instead", - ), ], ) @pytest.mark.parametrize( @@ -391,22 +375,6 @@ def test_check_if_deprecated_constant( DeprecatedConstantEnum(TestDeprecatedConstantEnum.TEST, "2099.1"), " which will be removed in HA Core 2099.1. Use TestDeprecatedConstantEnum.TEST instead", ), - ( - ("value", "NEW_CONSTANT", None), - ". Use NEW_CONSTANT instead", - ), - ( - (1, "NEW_CONSTANT", "2099.1"), - " which will be removed in HA Core 2099.1. Use NEW_CONSTANT instead", - ), - ( - (TestDeprecatedConstantEnum.TEST, None), - ". Use TestDeprecatedConstantEnum.TEST instead", - ), - ( - (TestDeprecatedConstantEnum.TEST, "2099.1"), - " which will be removed in HA Core 2099.1. Use TestDeprecatedConstantEnum.TEST instead", - ), ], ) @pytest.mark.parametrize( From d600b7680174854a41c7c37129434457744acc6b Mon Sep 17 00:00:00 2001 From: "J. Nick Koston" Date: Thu, 4 Jan 2024 02:21:52 -1000 Subject: [PATCH 13/26] Fix missing backwards compatibility layer for water_heater supported_features (#107091) --- .../components/water_heater/__init__.py | 4 ++-- tests/components/water_heater/test_init.py | 24 ++++++++++++------- 2 files changed, 18 insertions(+), 10 deletions(-) diff --git a/homeassistant/components/water_heater/__init__.py b/homeassistant/components/water_heater/__init__.py index f274441690067b..e5cf2cc2d3ce59 100644 --- a/homeassistant/components/water_heater/__init__.py +++ b/homeassistant/components/water_heater/__init__.py @@ -241,7 +241,7 @@ def capability_attributes(self) -> Mapping[str, Any]: ), } - if WaterHeaterEntityFeature.OPERATION_MODE in self.supported_features: + if WaterHeaterEntityFeature.OPERATION_MODE in self.supported_features_compat: data[ATTR_OPERATION_LIST] = self.operation_list return data @@ -277,7 +277,7 @@ def state_attributes(self) -> dict[str, Any]: ), } - supported_features = self.supported_features + supported_features = self.supported_features_compat if WaterHeaterEntityFeature.OPERATION_MODE in supported_features: data[ATTR_OPERATION_MODE] = self.current_operation diff --git a/tests/components/water_heater/test_init.py b/tests/components/water_heater/test_init.py index 0d33f3a9e936a4..861be19234088f 100644 --- a/tests/components/water_heater/test_init.py +++ b/tests/components/water_heater/test_init.py @@ -8,10 +8,13 @@ from homeassistant.components import water_heater from homeassistant.components.water_heater import ( + ATTR_OPERATION_LIST, + ATTR_OPERATION_MODE, SET_TEMPERATURE_SCHEMA, WaterHeaterEntity, WaterHeaterEntityFeature, ) +from homeassistant.const import UnitOfTemperature from homeassistant.core import HomeAssistant from tests.common import async_mock_service, import_and_test_deprecated_constant_enum @@ -117,21 +120,26 @@ def test_deprecated_constants( ) -def test_deprecated_supported_features_ints(caplog: pytest.LogCaptureFixture) -> None: +def test_deprecated_supported_features_ints( + hass: HomeAssistant, caplog: pytest.LogCaptureFixture +) -> None: """Test deprecated supported features ints.""" class MockWaterHeaterEntity(WaterHeaterEntity): - @property - def supported_features(self) -> int: - """Return supported features.""" - return 1 + _attr_operation_list = ["mode1", "mode2"] + _attr_temperature_unit = UnitOfTemperature.CELSIUS + _attr_current_operation = "mode1" + _attr_supported_features = WaterHeaterEntityFeature.OPERATION_MODE.value entity = MockWaterHeaterEntity() - assert entity.supported_features_compat is WaterHeaterEntityFeature(1) + entity.hass = hass + assert entity.supported_features_compat is WaterHeaterEntityFeature(2) assert "MockWaterHeaterEntity" in caplog.text assert "is using deprecated supported features values" in caplog.text assert "Instead it should use" in caplog.text - assert "WaterHeaterEntityFeature.TARGET_TEMPERATURE" in caplog.text + assert "WaterHeaterEntityFeature.OPERATION_MODE" in caplog.text caplog.clear() - assert entity.supported_features_compat is WaterHeaterEntityFeature(1) + assert entity.supported_features_compat is WaterHeaterEntityFeature(2) assert "is using deprecated supported features values" not in caplog.text + assert entity.state_attributes[ATTR_OPERATION_MODE] == "mode1" + assert entity.capability_attributes[ATTR_OPERATION_LIST] == ["mode1", "mode2"] From a7aa5c0e522a0c01bfc6aa152035e4a974149702 Mon Sep 17 00:00:00 2001 From: "J. Nick Koston" Date: Thu, 4 Jan 2024 02:45:47 -1000 Subject: [PATCH 14/26] Bump habluetooth to 2.0.2 (#107097) --- .../components/bluetooth/manifest.json | 2 +- homeassistant/package_constraints.txt | 2 +- requirements_all.txt | 2 +- requirements_test_all.txt | 2 +- tests/components/bluetooth/test_wrappers.py | 32 +++++++++++++++++++ 5 files changed, 36 insertions(+), 4 deletions(-) diff --git a/homeassistant/components/bluetooth/manifest.json b/homeassistant/components/bluetooth/manifest.json index c5dec12fe40143..7308f3a83ff718 100644 --- a/homeassistant/components/bluetooth/manifest.json +++ b/homeassistant/components/bluetooth/manifest.json @@ -20,6 +20,6 @@ "bluetooth-auto-recovery==1.2.3", "bluetooth-data-tools==1.19.0", "dbus-fast==2.21.0", - "habluetooth==2.0.1" + "habluetooth==2.0.2" ] } diff --git a/homeassistant/package_constraints.txt b/homeassistant/package_constraints.txt index 0f069a0e0b52dc..c5715c1c155a2f 100644 --- a/homeassistant/package_constraints.txt +++ b/homeassistant/package_constraints.txt @@ -24,7 +24,7 @@ dbus-fast==2.21.0 fnv-hash-fast==0.5.0 ha-av==10.1.1 ha-ffmpeg==3.1.0 -habluetooth==2.0.1 +habluetooth==2.0.2 hass-nabucasa==0.75.1 hassil==1.5.1 home-assistant-bluetooth==1.11.0 diff --git a/requirements_all.txt b/requirements_all.txt index 11d12f195a8607..7f5ef2a7b427f9 100644 --- a/requirements_all.txt +++ b/requirements_all.txt @@ -998,7 +998,7 @@ ha-philipsjs==3.1.1 habitipy==0.2.0 # homeassistant.components.bluetooth -habluetooth==2.0.1 +habluetooth==2.0.2 # homeassistant.components.cloud hass-nabucasa==0.75.1 diff --git a/requirements_test_all.txt b/requirements_test_all.txt index 767511cc9583fa..0d488cb865436a 100644 --- a/requirements_test_all.txt +++ b/requirements_test_all.txt @@ -803,7 +803,7 @@ ha-philipsjs==3.1.1 habitipy==0.2.0 # homeassistant.components.bluetooth -habluetooth==2.0.1 +habluetooth==2.0.2 # homeassistant.components.cloud hass-nabucasa==0.75.1 diff --git a/tests/components/bluetooth/test_wrappers.py b/tests/components/bluetooth/test_wrappers.py index cc837f381d4911..e3531a574471f6 100644 --- a/tests/components/bluetooth/test_wrappers.py +++ b/tests/components/bluetooth/test_wrappers.py @@ -367,6 +367,38 @@ async def connect(self, *args, **kwargs): cancel_hci1() +async def test_passing_subclassed_str_as_address( + hass: HomeAssistant, + two_adapters: None, + enable_bluetooth: None, + install_bleak_catcher, +) -> None: + """Ensure the client wrapper can handle a subclassed str as the address.""" + _, cancel_hci0, cancel_hci1 = _generate_scanners_with_fake_devices(hass) + + class SubclassedStr(str): + pass + + address = SubclassedStr("00:00:00:00:00:01") + client = bleak.BleakClient(address) + + class FakeBleakClient(BaseFakeBleakClient): + """Fake bleak client.""" + + async def connect(self, *args, **kwargs): + """Connect.""" + return True + + with patch( + "habluetooth.wrappers.get_platform_client_backend_type", + return_value=FakeBleakClient, + ): + assert await client.connect() is True + + cancel_hci0() + cancel_hci1() + + async def test_raise_after_shutdown( hass: HomeAssistant, two_adapters: None, From b8576b8091339cc7d485cc3d5166b83fd0d7108e Mon Sep 17 00:00:00 2001 From: Erik Montnemery Date: Fri, 5 Jan 2024 11:46:45 +0100 Subject: [PATCH 15/26] Include deprecated constants in wildcard imports (#107114) --- .../components/alarm_control_panel/__init__.py | 17 +++++++++++------ .../components/alarm_control_panel/const.py | 12 ++++++++---- .../components/automation/__init__.py | 13 +++++++++---- .../components/binary_sensor/__init__.py | 13 +++++++++---- homeassistant/components/camera/__init__.py | 13 +++++++++---- homeassistant/components/camera/const.py | 8 ++++++-- homeassistant/components/climate/__init__.py | 17 +++++++++++------ homeassistant/components/climate/const.py | 8 ++++++-- homeassistant/components/cover/__init__.py | 13 +++++++++---- .../components/device_tracker/__init__.py | 17 +++++++++++------ .../components/device_tracker/const.py | 12 ++++++++---- homeassistant/components/fan/__init__.py | 13 +++++++++---- .../components/humidifier/__init__.py | 17 +++++++++++------ homeassistant/components/humidifier/const.py | 8 ++++++-- homeassistant/components/lock/__init__.py | 13 +++++++++---- homeassistant/components/number/const.py | 12 ++++++++---- homeassistant/components/remote/__init__.py | 13 +++++++++---- homeassistant/components/sensor/__init__.py | 17 +++++++++++------ homeassistant/components/sensor/const.py | 12 ++++++++---- homeassistant/components/siren/__init__.py | 17 +++++++++++------ homeassistant/components/siren/const.py | 8 ++++++-- homeassistant/components/switch/__init__.py | 13 +++++++++---- .../components/water_heater/__init__.py | 13 +++++++++---- homeassistant/const.py | 14 ++++++++------ homeassistant/core.py | 14 +++++++++----- homeassistant/data_entry_flow.py | 13 +++++++++---- homeassistant/helpers/deprecation.py | 18 +++++++++++++++--- homeassistant/helpers/device_registry.py | 13 +++++++++---- pyproject.toml | 6 +++++- tests/common.py | 14 ++++++++++++-- .../alarm_control_panel/test_init.py | 11 ++++++++++- tests/components/automation/test_init.py | 6 ++++++ tests/components/binary_sensor/test_init.py | 6 ++++++ tests/components/camera/test_init.py | 11 ++++++++++- tests/components/climate/test_init.py | 10 ++++++++++ tests/components/cover/test_init.py | 7 ++++++- tests/components/device_tracker/test_init.py | 10 ++++++++++ tests/components/fan/test_init.py | 7 ++++++- tests/components/humidifier/test_init.py | 11 ++++++++++- tests/components/lock/test_init.py | 7 ++++++- tests/components/number/test_const.py | 7 ++++++- tests/components/remote/test_init.py | 11 ++++++++++- tests/components/sensor/test_init.py | 10 ++++++++++ tests/components/siren/test_init.py | 11 ++++++++++- tests/components/switch/test_init.py | 11 ++++++++++- tests/components/water_heater/test_init.py | 11 ++++++++++- tests/helpers/test_deprecation.py | 6 +++--- tests/helpers/test_device_registry.py | 6 ++++++ tests/test_const.py | 6 ++++++ tests/test_core.py | 6 ++++++ tests/test_data_entry_flow.py | 11 ++++++++++- .../test_constant_deprecation/__init__.py | 2 +- 52 files changed, 438 insertions(+), 137 deletions(-) diff --git a/homeassistant/components/alarm_control_panel/__init__.py b/homeassistant/components/alarm_control_panel/__init__.py index 9c53f2b7fd0b2d..45e1d63e0c2d15 100644 --- a/homeassistant/components/alarm_control_panel/__init__.py +++ b/homeassistant/components/alarm_control_panel/__init__.py @@ -24,6 +24,7 @@ import homeassistant.helpers.config_validation as cv from homeassistant.helpers.config_validation import make_entity_service_schema from homeassistant.helpers.deprecation import ( + all_with_deprecated_constants, check_if_deprecated_constant, dir_with_deprecated_constants, ) @@ -52,12 +53,6 @@ else: from homeassistant.backports.functools import cached_property -# As we import constants of the cost module here, we need to add the following -# functions to check for deprecated constants again -# Both can be removed if no deprecated constant are in this module anymore -__getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) -__dir__ = partial(dir_with_deprecated_constants, module_globals=globals()) - _LOGGER: Final = logging.getLogger(__name__) SCAN_INTERVAL: Final = timedelta(seconds=30) @@ -249,3 +244,13 @@ def state_attributes(self) -> dict[str, Any] | None: ATTR_CHANGED_BY: self.changed_by, ATTR_CODE_ARM_REQUIRED: self.code_arm_required, } + + +# As we import constants of the const module here, we need to add the following +# functions to check for deprecated constants again +# These can be removed if no deprecated constant are in this module anymore +__getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) +__dir__ = partial( + dir_with_deprecated_constants, module_globals_keys=[*globals().keys()] +) +__all__ = all_with_deprecated_constants(globals()) diff --git a/homeassistant/components/alarm_control_panel/const.py b/homeassistant/components/alarm_control_panel/const.py index 90bbcba1314b5f..fe4be649e195f9 100644 --- a/homeassistant/components/alarm_control_panel/const.py +++ b/homeassistant/components/alarm_control_panel/const.py @@ -5,6 +5,7 @@ from homeassistant.helpers.deprecation import ( DeprecatedConstantEnum, + all_with_deprecated_constants, check_if_deprecated_constant, dir_with_deprecated_constants, ) @@ -60,10 +61,6 @@ class AlarmControlPanelEntityFeature(IntFlag): AlarmControlPanelEntityFeature.ARM_VACATION, "2025.1" ) -# Both can be removed if no deprecated constant are in this module anymore -__getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) -__dir__ = partial(dir_with_deprecated_constants, module_globals=globals()) - CONDITION_TRIGGERED: Final = "is_triggered" CONDITION_DISARMED: Final = "is_disarmed" CONDITION_ARMED_HOME: Final = "is_armed_home" @@ -71,3 +68,10 @@ class AlarmControlPanelEntityFeature(IntFlag): CONDITION_ARMED_NIGHT: Final = "is_armed_night" CONDITION_ARMED_VACATION: Final = "is_armed_vacation" CONDITION_ARMED_CUSTOM_BYPASS: Final = "is_armed_custom_bypass" + +# These can be removed if no deprecated constant are in this module anymore +__getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) +__dir__ = partial( + dir_with_deprecated_constants, module_globals_keys=[*globals().keys()] +) +__all__ = all_with_deprecated_constants(globals()) diff --git a/homeassistant/components/automation/__init__.py b/homeassistant/components/automation/__init__.py index 4e6fa477ed2bac..efad44b15ef374 100644 --- a/homeassistant/components/automation/__init__.py +++ b/homeassistant/components/automation/__init__.py @@ -58,6 +58,7 @@ import homeassistant.helpers.config_validation as cv from homeassistant.helpers.deprecation import ( DeprecatedConstant, + all_with_deprecated_constants, check_if_deprecated_constant, dir_with_deprecated_constants, ) @@ -147,10 +148,6 @@ def __call__(self, variables: Mapping[str, Any] | None = None) -> bool: TriggerInfo, "TriggerInfo", "2025.1" ) -# Both can be removed if no deprecated constant are in this module anymore -__getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) -__dir__ = partial(dir_with_deprecated_constants, module_globals=globals()) - @bind_hass def is_on(hass: HomeAssistant, entity_id: str) -> bool: @@ -1108,3 +1105,11 @@ def websocket_config( "config": automation.raw_config, }, ) + + +# These can be removed if no deprecated constant are in this module anymore +__getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) +__dir__ = partial( + dir_with_deprecated_constants, module_globals_keys=[*globals().keys()] +) +__all__ = all_with_deprecated_constants(globals()) diff --git a/homeassistant/components/binary_sensor/__init__.py b/homeassistant/components/binary_sensor/__init__.py index 3a32a1afb57c51..0618548941948b 100644 --- a/homeassistant/components/binary_sensor/__init__.py +++ b/homeassistant/components/binary_sensor/__init__.py @@ -19,6 +19,7 @@ ) from homeassistant.helpers.deprecation import ( DeprecatedConstantEnum, + all_with_deprecated_constants, check_if_deprecated_constant, dir_with_deprecated_constants, ) @@ -218,10 +219,6 @@ class BinarySensorDeviceClass(StrEnum): BinarySensorDeviceClass.WINDOW, "2025.1" ) -# Both can be removed if no deprecated constant are in this module anymore -__getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) -__dir__ = partial(dir_with_deprecated_constants, module_globals=globals()) - # mypy: disallow-any-generics @@ -303,3 +300,11 @@ def state(self) -> Literal["on", "off"] | None: if (is_on := self.is_on) is None: return None return STATE_ON if is_on else STATE_OFF + + +# These can be removed if no deprecated constant are in this module anymore +__getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) +__dir__ = partial( + dir_with_deprecated_constants, module_globals_keys=[*globals().keys()] +) +__all__ = all_with_deprecated_constants(globals()) diff --git a/homeassistant/components/camera/__init__.py b/homeassistant/components/camera/__init__.py index 7a56292f7bb720..ce75f064d47658 100644 --- a/homeassistant/components/camera/__init__.py +++ b/homeassistant/components/camera/__init__.py @@ -54,6 +54,7 @@ ) from homeassistant.helpers.deprecation import ( DeprecatedConstantEnum, + all_with_deprecated_constants, check_if_deprecated_constant, dir_with_deprecated_constants, ) @@ -123,10 +124,6 @@ class CameraEntityFeature(IntFlag): CameraEntityFeature.STREAM, "2025.1" ) -# Both can be removed if no deprecated constant are in this module anymore -__getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) -__dir__ = partial(dir_with_deprecated_constants, module_globals=globals()) - RTSP_PREFIXES = {"rtsp://", "rtsps://", "rtmp://"} DEFAULT_CONTENT_TYPE: Final = "image/jpeg" @@ -1082,3 +1079,11 @@ async def async_handle_record_service( duration=service_call.data[CONF_DURATION], lookback=service_call.data[CONF_LOOKBACK], ) + + +# These can be removed if no deprecated constant are in this module anymore +__getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) +__dir__ = partial( + dir_with_deprecated_constants, module_globals_keys=[*globals().keys()] +) +__all__ = all_with_deprecated_constants(globals()) diff --git a/homeassistant/components/camera/const.py b/homeassistant/components/camera/const.py index da41c0b9fabc72..09c4c7c1fb21d1 100644 --- a/homeassistant/components/camera/const.py +++ b/homeassistant/components/camera/const.py @@ -5,6 +5,7 @@ from homeassistant.helpers.deprecation import ( DeprecatedConstantEnum, + all_with_deprecated_constants, check_if_deprecated_constant, dir_with_deprecated_constants, ) @@ -47,6 +48,9 @@ class StreamType(StrEnum): _DEPRECATED_STREAM_TYPE_WEB_RTC = DeprecatedConstantEnum(StreamType.WEB_RTC, "2025.1") -# Both can be removed if no deprecated constant are in this module anymore +# These can be removed if no deprecated constant are in this module anymore __getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) -__dir__ = partial(dir_with_deprecated_constants, module_globals=globals()) +__dir__ = partial( + dir_with_deprecated_constants, module_globals_keys=[*globals().keys()] +) +__all__ = all_with_deprecated_constants(globals()) diff --git a/homeassistant/components/climate/__init__.py b/homeassistant/components/climate/__init__.py index 78cb92944cbf86..c315765925f43c 100644 --- a/homeassistant/components/climate/__init__.py +++ b/homeassistant/components/climate/__init__.py @@ -28,6 +28,7 @@ make_entity_service_schema, ) from homeassistant.helpers.deprecation import ( + all_with_deprecated_constants, check_if_deprecated_constant, dir_with_deprecated_constants, ) @@ -141,12 +142,6 @@ ), ) -# As we import deprecated constants from the const module, we need to add these two functions -# otherwise this module will be logged for using deprecated constants and not the custom component -# Both can be removed if no deprecated constant are in this module anymore -__getattr__ = ft.partial(check_if_deprecated_constant, module_globals=globals()) -__dir__ = ft.partial(dir_with_deprecated_constants, module_globals=globals()) - # mypy: disallow-any-generics @@ -734,3 +729,13 @@ async def async_service_temperature_set( kwargs[value] = temp await entity.async_set_temperature(**kwargs) + + +# As we import deprecated constants from the const module, we need to add these two functions +# otherwise this module will be logged for using deprecated constants and not the custom component +# These can be removed if no deprecated constant are in this module anymore +__getattr__ = ft.partial(check_if_deprecated_constant, module_globals=globals()) +__dir__ = ft.partial( + dir_with_deprecated_constants, module_globals_keys=[*globals().keys()] +) +__all__ = all_with_deprecated_constants(globals()) diff --git a/homeassistant/components/climate/const.py b/homeassistant/components/climate/const.py index 615dc7d48dde5f..9c9153d9f63204 100644 --- a/homeassistant/components/climate/const.py +++ b/homeassistant/components/climate/const.py @@ -5,6 +5,7 @@ from homeassistant.helpers.deprecation import ( DeprecatedConstantEnum, + all_with_deprecated_constants, check_if_deprecated_constant, dir_with_deprecated_constants, ) @@ -188,6 +189,9 @@ class ClimateEntityFeature(IntFlag): ClimateEntityFeature.AUX_HEAT, "2025.1" ) -# Both can be removed if no deprecated constant are in this module anymore +# These can be removed if no deprecated constant are in this module anymore __getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) -__dir__ = partial(dir_with_deprecated_constants, module_globals=globals()) +__dir__ = partial( + dir_with_deprecated_constants, module_globals_keys=[*globals().keys()] +) +__all__ = all_with_deprecated_constants(globals()) diff --git a/homeassistant/components/cover/__init__.py b/homeassistant/components/cover/__init__.py index 3e438fb4ca1303..945585de522990 100644 --- a/homeassistant/components/cover/__init__.py +++ b/homeassistant/components/cover/__init__.py @@ -34,6 +34,7 @@ ) from homeassistant.helpers.deprecation import ( DeprecatedConstantEnum, + all_with_deprecated_constants, check_if_deprecated_constant, dir_with_deprecated_constants, ) @@ -143,10 +144,6 @@ class CoverEntityFeature(IntFlag): CoverEntityFeature.SET_TILT_POSITION, "2025.1" ) -# Both can be removed if no deprecated constant are in this module anymore -__getattr__ = ft.partial(check_if_deprecated_constant, module_globals=globals()) -__dir__ = ft.partial(dir_with_deprecated_constants, module_globals=globals()) - ATTR_CURRENT_POSITION = "current_position" ATTR_CURRENT_TILT_POSITION = "current_tilt_position" ATTR_POSITION = "position" @@ -493,3 +490,11 @@ def _get_toggle_function( if self._cover_is_last_toggle_direction_open: return fns["close"] return fns["open"] + + +# These can be removed if no deprecated constant are in this module anymore +__getattr__ = ft.partial(check_if_deprecated_constant, module_globals=globals()) +__dir__ = ft.partial( + dir_with_deprecated_constants, module_globals_keys=[*globals().keys()] +) +__all__ = all_with_deprecated_constants(globals()) diff --git a/homeassistant/components/device_tracker/__init__.py b/homeassistant/components/device_tracker/__init__.py index b5ad4660cde215..adcc90cccbfe26 100644 --- a/homeassistant/components/device_tracker/__init__.py +++ b/homeassistant/components/device_tracker/__init__.py @@ -6,6 +6,7 @@ from homeassistant.const import ATTR_GPS_ACCURACY, STATE_HOME # noqa: F401 from homeassistant.core import HomeAssistant from homeassistant.helpers.deprecation import ( + all_with_deprecated_constants, check_if_deprecated_constant, dir_with_deprecated_constants, ) @@ -57,12 +58,6 @@ see, ) -# As we import deprecated constants from the const module, we need to add these two functions -# otherwise this module will be logged for using deprecated constants and not the custom component -# Both can be removed if no deprecated constant are in this module anymore -__getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) -__dir__ = partial(dir_with_deprecated_constants, module_globals=globals()) - @bind_hass def is_on(hass: HomeAssistant, entity_id: str) -> bool: @@ -83,3 +78,13 @@ async def async_setup(hass: HomeAssistant, config: ConfigType) -> bool: await async_setup_legacy_integration(hass, config) return True + + +# As we import deprecated constants from the const module, we need to add these two functions +# otherwise this module will be logged for using deprecated constants and not the custom component +# These can be removed if no deprecated constant are in this module anymore +__getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) +__dir__ = partial( + dir_with_deprecated_constants, module_globals_keys=[*globals().keys()] +) +__all__ = all_with_deprecated_constants(globals()) diff --git a/homeassistant/components/device_tracker/const.py b/homeassistant/components/device_tracker/const.py index 10c16e09107026..67a90ab0f9522f 100644 --- a/homeassistant/components/device_tracker/const.py +++ b/homeassistant/components/device_tracker/const.py @@ -9,6 +9,7 @@ from homeassistant.helpers.deprecation import ( DeprecatedConstantEnum, + all_with_deprecated_constants, check_if_deprecated_constant, dir_with_deprecated_constants, ) @@ -44,10 +45,6 @@ class SourceType(StrEnum): SourceType.BLUETOOTH_LE, "2025.1" ) -# Both can be removed if no deprecated constant are in this module anymore -__getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) -__dir__ = partial(dir_with_deprecated_constants, module_globals=globals()) - CONF_SCAN_INTERVAL: Final = "interval_seconds" SCAN_INTERVAL: Final = timedelta(seconds=12) @@ -71,3 +68,10 @@ class SourceType(StrEnum): ATTR_IP: Final = "ip" CONNECTED_DEVICE_REGISTERED: Final = "device_tracker_connected_device_registered" + +# These can be removed if no deprecated constant are in this module anymore +__getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) +__dir__ = partial( + dir_with_deprecated_constants, module_globals_keys=[*globals().keys()] +) +__all__ = all_with_deprecated_constants(globals()) diff --git a/homeassistant/components/fan/__init__.py b/homeassistant/components/fan/__init__.py index dedaedfe60002f..c35d828e3984e4 100644 --- a/homeassistant/components/fan/__init__.py +++ b/homeassistant/components/fan/__init__.py @@ -26,6 +26,7 @@ ) from homeassistant.helpers.deprecation import ( DeprecatedConstantEnum, + all_with_deprecated_constants, check_if_deprecated_constant, dir_with_deprecated_constants, ) @@ -76,10 +77,6 @@ class FanEntityFeature(IntFlag): FanEntityFeature.PRESET_MODE, "2025.1" ) -# Both can be removed if no deprecated constant are in this module anymore -__getattr__ = ft.partial(check_if_deprecated_constant, module_globals=globals()) -__dir__ = ft.partial(dir_with_deprecated_constants, module_globals=globals()) - SERVICE_INCREASE_SPEED = "increase_speed" SERVICE_DECREASE_SPEED = "decrease_speed" SERVICE_OSCILLATE = "oscillate" @@ -471,3 +468,11 @@ def preset_modes(self) -> list[str] | None: if hasattr(self, "_attr_preset_modes"): return self._attr_preset_modes return None + + +# These can be removed if no deprecated constant are in this module anymore +__getattr__ = ft.partial(check_if_deprecated_constant, module_globals=globals()) +__dir__ = ft.partial( + dir_with_deprecated_constants, module_globals_keys=[*globals().keys()] +) +__all__ = all_with_deprecated_constants(globals()) diff --git a/homeassistant/components/humidifier/__init__.py b/homeassistant/components/humidifier/__init__.py index 0d8f2e29561640..65a8cd9d1d05d2 100644 --- a/homeassistant/components/humidifier/__init__.py +++ b/homeassistant/components/humidifier/__init__.py @@ -24,6 +24,7 @@ PLATFORM_SCHEMA_BASE, ) from homeassistant.helpers.deprecation import ( + all_with_deprecated_constants, check_if_deprecated_constant, dir_with_deprecated_constants, ) @@ -81,12 +82,6 @@ class HumidifierDeviceClass(StrEnum): # use the HumidifierDeviceClass enum instead. DEVICE_CLASSES = [cls.value for cls in HumidifierDeviceClass] -# As we import deprecated constants from the const module, we need to add these two functions -# otherwise this module will be logged for using deprecated constants and not the custom component -# Both can be removed if no deprecated constant are in this module anymore -__getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) -__dir__ = partial(dir_with_deprecated_constants, module_globals=globals()) - # mypy: disallow-any-generics @@ -293,3 +288,13 @@ def supported_features_compat(self) -> HumidifierEntityFeature: self._report_deprecated_supported_features_values(new_features) return new_features return features + + +# As we import deprecated constants from the const module, we need to add these two functions +# otherwise this module will be logged for using deprecated constants and not the custom component +# These can be removed if no deprecated constant are in this module anymore +__getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) +__dir__ = partial( + dir_with_deprecated_constants, module_globals_keys=[*globals().keys()] +) +__all__ = all_with_deprecated_constants(globals()) diff --git a/homeassistant/components/humidifier/const.py b/homeassistant/components/humidifier/const.py index a1a219ddce7649..66ac0fcf18d36a 100644 --- a/homeassistant/components/humidifier/const.py +++ b/homeassistant/components/humidifier/const.py @@ -5,6 +5,7 @@ from homeassistant.helpers.deprecation import ( DeprecatedConstant, DeprecatedConstantEnum, + all_with_deprecated_constants, check_if_deprecated_constant, dir_with_deprecated_constants, ) @@ -66,6 +67,9 @@ class HumidifierEntityFeature(IntFlag): HumidifierEntityFeature.MODES, "2025.1" ) -# Both can be removed if no deprecated constant are in this module anymore +# These can be removed if no deprecated constant are in this module anymore __getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) -__dir__ = partial(dir_with_deprecated_constants, module_globals=globals()) +__dir__ = partial( + dir_with_deprecated_constants, module_globals_keys=[*globals().keys()] +) +__all__ = all_with_deprecated_constants(globals()) diff --git a/homeassistant/components/lock/__init__.py b/homeassistant/components/lock/__init__.py index a9370f8d092663..a4e7c4b7d1ad48 100644 --- a/homeassistant/components/lock/__init__.py +++ b/homeassistant/components/lock/__init__.py @@ -33,6 +33,7 @@ ) from homeassistant.helpers.deprecation import ( DeprecatedConstantEnum, + all_with_deprecated_constants, check_if_deprecated_constant, dir_with_deprecated_constants, ) @@ -70,10 +71,6 @@ class LockEntityFeature(IntFlag): # Please use the LockEntityFeature enum instead. _DEPRECATED_SUPPORT_OPEN = DeprecatedConstantEnum(LockEntityFeature.OPEN, "2025.1") -# Both can be removed if no deprecated constant are in this module anymore -__getattr__ = ft.partial(check_if_deprecated_constant, module_globals=globals()) -__dir__ = ft.partial(dir_with_deprecated_constants, module_globals=globals()) - PROP_TO_ATTR = {"changed_by": ATTR_CHANGED_BY, "code_format": ATTR_CODE_FORMAT} # mypy: disallow-any-generics @@ -315,3 +312,11 @@ def _async_read_entity_options(self) -> None: return self._lock_option_default_code = "" + + +# These can be removed if no deprecated constant are in this module anymore +__getattr__ = ft.partial(check_if_deprecated_constant, module_globals=globals()) +__dir__ = ft.partial( + dir_with_deprecated_constants, module_globals_keys=[*globals().keys()] +) +__all__ = all_with_deprecated_constants(globals()) diff --git a/homeassistant/components/number/const.py b/homeassistant/components/number/const.py index 55d22c86648935..a2d7c066af75f8 100644 --- a/homeassistant/components/number/const.py +++ b/homeassistant/components/number/const.py @@ -38,6 +38,7 @@ ) from homeassistant.helpers.deprecation import ( DeprecatedConstantEnum, + all_with_deprecated_constants, check_if_deprecated_constant, dir_with_deprecated_constants, ) @@ -70,10 +71,6 @@ class NumberMode(StrEnum): _DEPRECATED_MODE_BOX: Final = DeprecatedConstantEnum(NumberMode.BOX, "2025.1") _DEPRECATED_MODE_SLIDER: Final = DeprecatedConstantEnum(NumberMode.SLIDER, "2025.1") -# Both can be removed if no deprecated constant are in this module anymore -__getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) -__dir__ = partial(dir_with_deprecated_constants, module_globals=globals()) - class NumberDeviceClass(StrEnum): """Device class for numbers.""" @@ -481,3 +478,10 @@ class NumberDeviceClass(StrEnum): UNIT_CONVERTERS: dict[str, type[BaseUnitConverter]] = { NumberDeviceClass.TEMPERATURE: TemperatureConverter, } + +# These can be removed if no deprecated constant are in this module anymore +__getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) +__dir__ = partial( + dir_with_deprecated_constants, module_globals_keys=[*globals().keys()] +) +__all__ = all_with_deprecated_constants(globals()) diff --git a/homeassistant/components/remote/__init__.py b/homeassistant/components/remote/__init__.py index 7e9ebfe12b9f5e..c5facb9785cc64 100644 --- a/homeassistant/components/remote/__init__.py +++ b/homeassistant/components/remote/__init__.py @@ -27,6 +27,7 @@ ) from homeassistant.helpers.deprecation import ( DeprecatedConstantEnum, + all_with_deprecated_constants, check_if_deprecated_constant, dir_with_deprecated_constants, ) @@ -92,10 +93,6 @@ class RemoteEntityFeature(IntFlag): ) -# Both can be removed if no deprecated constant are in this module anymore -__getattr__ = ft.partial(check_if_deprecated_constant, module_globals=globals()) -__dir__ = ft.partial(dir_with_deprecated_constants, module_globals=globals()) - REMOTE_SERVICE_ACTIVITY_SCHEMA = make_entity_service_schema( {vol.Optional(ATTR_ACTIVITY): cv.string} ) @@ -262,3 +259,11 @@ async def async_delete_command(self, **kwargs: Any) -> None: await self.hass.async_add_executor_job( ft.partial(self.delete_command, **kwargs) ) + + +# These can be removed if no deprecated constant are in this module anymore +__getattr__ = ft.partial(check_if_deprecated_constant, module_globals=globals()) +__dir__ = ft.partial( + dir_with_deprecated_constants, module_globals_keys=[*globals().keys()] +) +__all__ = all_with_deprecated_constants(globals()) diff --git a/homeassistant/components/sensor/__init__.py b/homeassistant/components/sensor/__init__.py index d7c5cddc5db696..6498b92b03e95b 100644 --- a/homeassistant/components/sensor/__init__.py +++ b/homeassistant/components/sensor/__init__.py @@ -59,6 +59,7 @@ PLATFORM_SCHEMA_BASE, ) from homeassistant.helpers.deprecation import ( + all_with_deprecated_constants, check_if_deprecated_constant, dir_with_deprecated_constants, ) @@ -120,12 +121,6 @@ "SensorStateClass", ] -# As we import deprecated constants from the const module, we need to add these two functions -# otherwise this module will be logged for using deprecated constants and not the custom component -# Both can be removed if no deprecated constant are in this module anymore -__getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) -__dir__ = partial(dir_with_deprecated_constants, module_globals=globals()) - # mypy: disallow-any-generics @@ -955,3 +950,13 @@ def async_rounded_state(hass: HomeAssistant, entity_id: str, state: State) -> st value = f"{numerical_value:z.{precision}f}" return value + + +# As we import deprecated constants from the const module, we need to add these two functions +# otherwise this module will be logged for using deprecated constants and not the custom component +# These can be removed if no deprecated constant are in this module anymore +__getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) +__dir__ = partial( + dir_with_deprecated_constants, module_globals_keys=[*globals().keys()] +) +__all__ = all_with_deprecated_constants(globals()) diff --git a/homeassistant/components/sensor/const.py b/homeassistant/components/sensor/const.py index d57a09981efba7..b1cb120e3fee36 100644 --- a/homeassistant/components/sensor/const.py +++ b/homeassistant/components/sensor/const.py @@ -38,6 +38,7 @@ ) from homeassistant.helpers.deprecation import ( DeprecatedConstantEnum, + all_with_deprecated_constants, check_if_deprecated_constant, dir_with_deprecated_constants, ) @@ -468,10 +469,6 @@ class SensorStateClass(StrEnum): ) STATE_CLASSES: Final[list[str]] = [cls.value for cls in SensorStateClass] -# Both can be removed if no deprecated constant are in this module anymore -__getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) -__dir__ = partial(dir_with_deprecated_constants, module_globals=globals()) - UNIT_CONVERTERS: dict[SensorDeviceClass | str | None, type[BaseUnitConverter]] = { SensorDeviceClass.ATMOSPHERIC_PRESSURE: PressureConverter, SensorDeviceClass.CURRENT: ElectricCurrentConverter, @@ -631,3 +628,10 @@ class SensorStateClass(StrEnum): SensorDeviceClass.WEIGHT: {SensorStateClass.MEASUREMENT}, SensorDeviceClass.WIND_SPEED: {SensorStateClass.MEASUREMENT}, } + +# These can be removed if no deprecated constant are in this module anymore +__getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) +__dir__ = partial( + dir_with_deprecated_constants, module_globals_keys=[*globals().keys()] +) +__all__ = all_with_deprecated_constants(globals()) diff --git a/homeassistant/components/siren/__init__.py b/homeassistant/components/siren/__init__.py index 29ad238ac00474..fb41d5f7b48845 100644 --- a/homeassistant/components/siren/__init__.py +++ b/homeassistant/components/siren/__init__.py @@ -17,6 +17,7 @@ PLATFORM_SCHEMA_BASE, ) from homeassistant.helpers.deprecation import ( + all_with_deprecated_constants, check_if_deprecated_constant, dir_with_deprecated_constants, ) @@ -53,12 +54,6 @@ vol.Optional(ATTR_VOLUME_LEVEL): cv.small_float, } -# As we import deprecated constants from the const module, we need to add these two functions -# otherwise this module will be logged for using deprecated constants and not the custom component -# Both can be removed if no deprecated constant are in this module anymore -__getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) -__dir__ = partial(dir_with_deprecated_constants, module_globals=globals()) - class SirenTurnOnServiceParameters(TypedDict, total=False): """Represent possible parameters to siren.turn_on service data dict type.""" @@ -218,3 +213,13 @@ def supported_features(self) -> SirenEntityFeature: self._report_deprecated_supported_features_values(new_features) return new_features return features + + +# As we import deprecated constants from the const module, we need to add these two functions +# otherwise this module will be logged for using deprecated constants and not the custom component +# These can be removed if no deprecated constant are in this module anymore +__getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) +__dir__ = partial( + dir_with_deprecated_constants, module_globals_keys=[*globals().keys()] +) +__all__ = all_with_deprecated_constants(globals()) diff --git a/homeassistant/components/siren/const.py b/homeassistant/components/siren/const.py index 50c3af61c8dd6e..9e46d8dc997b94 100644 --- a/homeassistant/components/siren/const.py +++ b/homeassistant/components/siren/const.py @@ -6,6 +6,7 @@ from homeassistant.helpers.deprecation import ( DeprecatedConstantEnum, + all_with_deprecated_constants, check_if_deprecated_constant, dir_with_deprecated_constants, ) @@ -47,6 +48,9 @@ class SirenEntityFeature(IntFlag): SirenEntityFeature.DURATION, "2025.1" ) -# Both can be removed if no deprecated constant are in this module anymore +# These can be removed if no deprecated constant are in this module anymore __getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) -__dir__ = partial(dir_with_deprecated_constants, module_globals=globals()) +__dir__ = partial( + dir_with_deprecated_constants, module_globals_keys=[*globals().keys()] +) +__all__ = all_with_deprecated_constants(globals()) diff --git a/homeassistant/components/switch/__init__.py b/homeassistant/components/switch/__init__.py index a318f763fcb70b..ce9b1477ad6a23 100644 --- a/homeassistant/components/switch/__init__.py +++ b/homeassistant/components/switch/__init__.py @@ -23,6 +23,7 @@ ) from homeassistant.helpers.deprecation import ( DeprecatedConstantEnum, + all_with_deprecated_constants, check_if_deprecated_constant, dir_with_deprecated_constants, ) @@ -66,10 +67,6 @@ class SwitchDeviceClass(StrEnum): SwitchDeviceClass.SWITCH, "2025.1" ) -# Both can be removed if no deprecated constant are in this module anymore -__getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) -__dir__ = partial(dir_with_deprecated_constants, module_globals=globals()) - # mypy: disallow-any-generics @@ -133,3 +130,11 @@ def device_class(self) -> SwitchDeviceClass | None: if hasattr(self, "entity_description"): return self.entity_description.device_class return None + + +# These can be removed if no deprecated constant are in this module anymore +__getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) +__dir__ = partial( + dir_with_deprecated_constants, module_globals_keys=[*globals().keys()] +) +__all__ = all_with_deprecated_constants(globals()) diff --git a/homeassistant/components/water_heater/__init__.py b/homeassistant/components/water_heater/__init__.py index e5cf2cc2d3ce59..82a853125ff1f5 100644 --- a/homeassistant/components/water_heater/__init__.py +++ b/homeassistant/components/water_heater/__init__.py @@ -30,6 +30,7 @@ ) from homeassistant.helpers.deprecation import ( DeprecatedConstantEnum, + all_with_deprecated_constants, check_if_deprecated_constant, dir_with_deprecated_constants, ) @@ -86,10 +87,6 @@ class WaterHeaterEntityFeature(IntFlag): WaterHeaterEntityFeature.AWAY_MODE, "2025.1" ) -# Both can be removed if no deprecated constant are in this module anymore -__getattr__ = ft.partial(check_if_deprecated_constant, module_globals=globals()) -__dir__ = ft.partial(dir_with_deprecated_constants, module_globals=globals()) - ATTR_MAX_TEMP = "max_temp" ATTR_MIN_TEMP = "min_temp" ATTR_AWAY_MODE = "away_mode" @@ -441,3 +438,11 @@ async def async_service_temperature_set( kwargs[value] = temp await entity.async_set_temperature(**kwargs) + + +# These can be removed if no deprecated constant are in this module anymore +__getattr__ = ft.partial(check_if_deprecated_constant, module_globals=globals()) +__dir__ = ft.partial( + dir_with_deprecated_constants, module_globals_keys=[*globals().keys()] +) +__all__ = all_with_deprecated_constants(globals()) diff --git a/homeassistant/const.py b/homeassistant/const.py index a6927aa8165aed..9e0505fadf333f 100644 --- a/homeassistant/const.py +++ b/homeassistant/const.py @@ -8,6 +8,7 @@ from .helpers.deprecation import ( DeprecatedConstant, DeprecatedConstantEnum, + all_with_deprecated_constants, check_if_deprecated_constant, dir_with_deprecated_constants, ) @@ -440,12 +441,6 @@ class Platform(StrEnum): "2025.1", ) - -# Both can be removed if no deprecated constant are in this module anymore -__getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) -__dir__ = partial(dir_with_deprecated_constants, module_globals=globals()) - - # #### STATES #### STATE_ON: Final = "on" STATE_OFF: Final = "off" @@ -1607,3 +1602,10 @@ class EntityCategory(StrEnum): FORMAT_DATE: Final = "%Y-%m-%d" FORMAT_TIME: Final = "%H:%M:%S" FORMAT_DATETIME: Final = f"{FORMAT_DATE} {FORMAT_TIME}" + +# These can be removed if no deprecated constant are in this module anymore +__getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) +__dir__ = partial( + dir_with_deprecated_constants, module_globals_keys=[*globals().keys()] +) +__all__ = all_with_deprecated_constants(globals()) diff --git a/homeassistant/core.py b/homeassistant/core.py index b15d393c63e237..e843481f79daf8 100644 --- a/homeassistant/core.py +++ b/homeassistant/core.py @@ -83,6 +83,7 @@ ) from .helpers.deprecation import ( DeprecatedConstantEnum, + all_with_deprecated_constants, check_if_deprecated_constant, dir_with_deprecated_constants, ) @@ -159,11 +160,6 @@ class ConfigSource(enum.StrEnum): _DEPRECATED_SOURCE_YAML = DeprecatedConstantEnum(ConfigSource.YAML, "2025.1") -# Both can be removed if no deprecated constant are in this module anymore -__getattr__ = functools.partial(check_if_deprecated_constant, module_globals=globals()) -__dir__ = functools.partial(dir_with_deprecated_constants, module_globals=globals()) - - # How long to wait until things that run on startup have to finish. TIMEOUT_EVENT_START = 15 @@ -2534,3 +2530,11 @@ async def async_save(self, data: dict[str, Any]) -> None: if self._original_unit_system: data["unit_system"] = self._original_unit_system return await super().async_save(data) + + +# These can be removed if no deprecated constant are in this module anymore +__getattr__ = functools.partial(check_if_deprecated_constant, module_globals=globals()) +__dir__ = functools.partial( + dir_with_deprecated_constants, module_globals_keys=[*globals().keys()] +) +__all__ = all_with_deprecated_constants(globals()) diff --git a/homeassistant/data_entry_flow.py b/homeassistant/data_entry_flow.py index 5c9c0ff1ce4d7f..63ba565582a8bf 100644 --- a/homeassistant/data_entry_flow.py +++ b/homeassistant/data_entry_flow.py @@ -17,6 +17,7 @@ from .exceptions import HomeAssistantError from .helpers.deprecation import ( DeprecatedConstantEnum, + all_with_deprecated_constants, check_if_deprecated_constant, dir_with_deprecated_constants, ) @@ -59,10 +60,6 @@ class FlowResultType(StrEnum): ) _DEPRECATED_RESULT_TYPE_MENU = DeprecatedConstantEnum(FlowResultType.MENU, "2025.1") -# Both can be removed if no deprecated constant are in this module anymore -__getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) -__dir__ = partial(dir_with_deprecated_constants, module_globals=globals()) - # Event that is fired when a flow is progressed via external or progress source. EVENT_DATA_ENTRY_FLOW_PROGRESSED = "data_entry_flow_progressed" @@ -700,3 +697,11 @@ def _create_abort_data( reason=reason, description_placeholders=description_placeholders, ) + + +# These can be removed if no deprecated constant are in this module anymore +__getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) +__dir__ = partial( + dir_with_deprecated_constants, module_globals_keys=[*globals().keys()] +) +__all__ = all_with_deprecated_constants(globals()) diff --git a/homeassistant/helpers/deprecation.py b/homeassistant/helpers/deprecation.py index 72b26e90b84275..18a42ce9bcf1d7 100644 --- a/homeassistant/helpers/deprecation.py +++ b/homeassistant/helpers/deprecation.py @@ -292,10 +292,22 @@ def check_if_deprecated_constant(name: str, module_globals: dict[str, Any]) -> A return value -def dir_with_deprecated_constants(module_globals: dict[str, Any]) -> list[str]: +def dir_with_deprecated_constants(module_globals_keys: list[str]) -> list[str]: """Return dir() with deprecated constants.""" - return list(module_globals) + [ + return module_globals_keys + [ name.removeprefix(_PREFIX_DEPRECATED) - for name in module_globals + for name in module_globals_keys + if name.startswith(_PREFIX_DEPRECATED) + ] + + +def all_with_deprecated_constants(module_globals: dict[str, Any]) -> list[str]: + """Generate a list for __all___ with deprecated constants.""" + # Iterate over a copy in case the globals dict is mutated by another thread + # while we loop over it. + module_globals_keys = list(module_globals) + return [itm for itm in module_globals_keys if not itm.startswith("_")] + [ + name.removeprefix(_PREFIX_DEPRECATED) + for name in module_globals_keys if name.startswith(_PREFIX_DEPRECATED) ] diff --git a/homeassistant/helpers/device_registry.py b/homeassistant/helpers/device_registry.py index bd509cb47ec24d..cfe3b78ebab3fe 100644 --- a/homeassistant/helpers/device_registry.py +++ b/homeassistant/helpers/device_registry.py @@ -24,6 +24,7 @@ from .debounce import Debouncer from .deprecation import ( DeprecatedConstantEnum, + all_with_deprecated_constants, check_if_deprecated_constant, dir_with_deprecated_constants, ) @@ -75,10 +76,6 @@ class DeviceEntryDisabler(StrEnum): ) _DEPRECATED_DISABLED_USER = DeprecatedConstantEnum(DeviceEntryDisabler.USER, "2025.1") -# Both can be removed if no deprecated constant are in this module anymore -__getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) -__dir__ = partial(dir_with_deprecated_constants, module_globals=globals()) - class DeviceInfo(TypedDict, total=False): """Entity device information for device registry.""" @@ -1113,3 +1110,11 @@ def _normalize_connections(connections: set[tuple[str, str]]) -> set[tuple[str, (key, format_mac(value)) if key == CONNECTION_NETWORK_MAC else (key, value) for key, value in connections } + + +# These can be removed if no deprecated constant are in this module anymore +__getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) +__dir__ = partial( + dir_with_deprecated_constants, module_globals_keys=[*globals().keys()] +) +__all__ = all_with_deprecated_constants(globals()) diff --git a/pyproject.toml b/pyproject.toml index ec313a5bcf6deb..2fc5f594157b0a 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -176,7 +176,8 @@ disable = [ "duplicate-bases", # PLE0241 "format-needs-mapping", # F502 "function-redefined", # F811 - "invalid-all-format", # PLE0605 + # Needed because ruff does not understand type of __all__ generated by a function + # "invalid-all-format", # PLE0605 "invalid-all-object", # PLE0604 "invalid-character-backspace", # PLE2510 "invalid-character-esc", # PLE2513 @@ -673,6 +674,9 @@ ignore = [ "COM819", "ISC001", "ISC002", + + # Disabled because ruff does not understand type of __all__ generated by a function + "PLE0605", ] [tool.ruff.flake8-import-conventions.extend-aliases] diff --git a/tests/common.py b/tests/common.py index b07788dc3d73a3..85193022e4f9a8 100644 --- a/tests/common.py +++ b/tests/common.py @@ -92,7 +92,7 @@ import homeassistant.util.yaml.loader as yaml_loader from tests.testing_config.custom_components.test_constant_deprecation import ( - import_deprecated_costant, + import_deprecated_constant, ) _LOGGER = logging.getLogger(__name__) @@ -1482,6 +1482,7 @@ def import_and_test_deprecated_constant_enum( - Assert value is the same as the replacement - Assert a warning is logged - Assert the deprecated constant is included in the modules.__dir__() + - Assert the deprecated constant is included in the modules.__all__() """ import_and_test_deprecated_constant( caplog, @@ -1507,8 +1508,9 @@ def import_and_test_deprecated_constant( - Assert value is the same as the replacement - Assert a warning is logged - Assert the deprecated constant is included in the modules.__dir__() + - Assert the deprecated constant is included in the modules.__all__() """ - value = import_deprecated_costant(module, constant_name) + value = import_deprecated_constant(module, constant_name) assert value == replacement assert ( module.__name__, @@ -1523,3 +1525,11 @@ def import_and_test_deprecated_constant( # verify deprecated constant is included in dir() assert constant_name in dir(module) + assert constant_name in module.__all__ + + +def help_test_all(module: ModuleType) -> None: + """Test module.__all__ is correctly set.""" + assert set(module.__all__) == { + itm for itm in module.__dir__() if not itm.startswith("_") + } diff --git a/tests/components/alarm_control_panel/test_init.py b/tests/components/alarm_control_panel/test_init.py index 1e6fce6def6250..42a532cbb1a196 100644 --- a/tests/components/alarm_control_panel/test_init.py +++ b/tests/components/alarm_control_panel/test_init.py @@ -6,7 +6,16 @@ from homeassistant.components import alarm_control_panel -from tests.common import import_and_test_deprecated_constant_enum +from tests.common import help_test_all, import_and_test_deprecated_constant_enum + + +@pytest.mark.parametrize( + "module", + [alarm_control_panel, alarm_control_panel.const], +) +def test_all(module: ModuleType) -> None: + """Test module.__all__ is correctly set.""" + help_test_all(module) @pytest.mark.parametrize( diff --git a/tests/components/automation/test_init.py b/tests/components/automation/test_init.py index 235ca48f095e42..6bb1b89259a61c 100644 --- a/tests/components/automation/test_init.py +++ b/tests/components/automation/test_init.py @@ -59,6 +59,7 @@ async_capture_events, async_fire_time_changed, async_mock_service, + help_test_all, import_and_test_deprecated_constant, mock_restore_cache, ) @@ -2569,6 +2570,11 @@ async def test_websocket_config( assert msg["error"]["code"] == "not_found" +def test_all() -> None: + """Test module.__all__ is correctly set.""" + help_test_all(automation) + + @pytest.mark.parametrize( ("constant_name", "replacement"), [ diff --git a/tests/components/binary_sensor/test_init.py b/tests/components/binary_sensor/test_init.py index 014722d94a4f36..6ca189113b9eaf 100644 --- a/tests/components/binary_sensor/test_init.py +++ b/tests/components/binary_sensor/test_init.py @@ -14,6 +14,7 @@ MockConfigEntry, MockModule, MockPlatform, + help_test_all, import_and_test_deprecated_constant_enum, mock_config_flow, mock_integration, @@ -197,6 +198,11 @@ async def async_setup_entry_platform( ) +def test_all() -> None: + """Test module.__all__ is correctly set.""" + help_test_all(binary_sensor) + + @pytest.mark.parametrize( "device_class", list(binary_sensor.BinarySensorDeviceClass), diff --git a/tests/components/camera/test_init.py b/tests/components/camera/test_init.py index 0e761f2f437230..f1e3a4fdef5bc1 100644 --- a/tests/components/camera/test_init.py +++ b/tests/components/camera/test_init.py @@ -27,7 +27,7 @@ from .common import EMPTY_8_6_JPEG, WEBRTC_ANSWER, mock_turbo_jpeg -from tests.common import import_and_test_deprecated_constant_enum +from tests.common import help_test_all, import_and_test_deprecated_constant_enum from tests.typing import ClientSessionGenerator, WebSocketGenerator STREAM_SOURCE = "rtsp://127.0.0.1/stream" @@ -962,6 +962,15 @@ async def test_use_stream_for_stills( assert await resp.read() == b"stream_keyframe_image" +@pytest.mark.parametrize( + "module", + [camera, camera.const], +) +def test_all(module: ModuleType) -> None: + """Test module.__all__ is correctly set.""" + help_test_all(module) + + @pytest.mark.parametrize( "enum", list(camera.const.StreamType), diff --git a/tests/components/climate/test_init.py b/tests/components/climate/test_init.py index 8fc82365c23ee5..89826c98086612 100644 --- a/tests/components/climate/test_init.py +++ b/tests/components/climate/test_init.py @@ -36,6 +36,7 @@ MockModule, MockPlatform, async_mock_service, + help_test_all, import_and_test_deprecated_constant, import_and_test_deprecated_constant_enum, mock_integration, @@ -157,6 +158,15 @@ def _create_tuples(enum: Enum, constant_prefix: str) -> list[tuple[Enum, str]]: return result +@pytest.mark.parametrize( + "module", + [climate, climate.const], +) +def test_all(module: ModuleType) -> None: + """Test module.__all__ is correctly set.""" + help_test_all(module) + + @pytest.mark.parametrize( ("enum", "constant_prefix"), _create_tuples(climate.ClimateEntityFeature, "SUPPORT_") diff --git a/tests/components/cover/test_init.py b/tests/components/cover/test_init.py index 1b08658d9839d3..480d1ef83aa571 100644 --- a/tests/components/cover/test_init.py +++ b/tests/components/cover/test_init.py @@ -16,7 +16,7 @@ from homeassistant.core import HomeAssistant from homeassistant.setup import async_setup_component -from tests.common import import_and_test_deprecated_constant_enum +from tests.common import help_test_all, import_and_test_deprecated_constant_enum async def test_services(hass: HomeAssistant, enable_custom_integrations: None) -> None: @@ -127,6 +127,11 @@ def _create_tuples(enum: Enum, constant_prefix: str) -> list[tuple[Enum, str]]: return result +def test_all() -> None: + """Test module.__all__ is correctly set.""" + help_test_all(cover) + + @pytest.mark.parametrize( ("enum", "constant_prefix"), _create_tuples(cover.CoverEntityFeature, "SUPPORT_") diff --git a/tests/components/device_tracker/test_init.py b/tests/components/device_tracker/test_init.py index 024187a33f6a9e..eb8fde8f0e2731 100644 --- a/tests/components/device_tracker/test_init.py +++ b/tests/components/device_tracker/test_init.py @@ -34,6 +34,7 @@ from tests.common import ( assert_setup_component, async_fire_time_changed, + help_test_all, import_and_test_deprecated_constant_enum, mock_registry, mock_restore_cache, @@ -685,6 +686,15 @@ def test_see_schema_allowing_ios_calls() -> None: ) +@pytest.mark.parametrize( + "module", + [device_tracker, device_tracker.const], +) +def test_all(module: ModuleType) -> None: + """Test module.__all__ is correctly set.""" + help_test_all(module) + + @pytest.mark.parametrize(("enum"), list(SourceType)) @pytest.mark.parametrize( "module", diff --git a/tests/components/fan/test_init.py b/tests/components/fan/test_init.py index 828c13b6f167de..1beea47c6fa845 100644 --- a/tests/components/fan/test_init.py +++ b/tests/components/fan/test_init.py @@ -15,7 +15,7 @@ import homeassistant.helpers.entity_registry as er from homeassistant.setup import async_setup_component -from tests.common import import_and_test_deprecated_constant_enum +from tests.common import help_test_all, import_and_test_deprecated_constant_enum from tests.testing_config.custom_components.test.fan import MockFan @@ -150,6 +150,11 @@ async def test_preset_mode_validation( assert exc.value.translation_key == "not_valid_preset_mode" +def test_all() -> None: + """Test module.__all__ is correctly set.""" + help_test_all(fan) + + @pytest.mark.parametrize(("enum"), list(fan.FanEntityFeature)) def test_deprecated_constants( caplog: pytest.LogCaptureFixture, diff --git a/tests/components/humidifier/test_init.py b/tests/components/humidifier/test_init.py index 24cf4b6d962939..3ef3fca8589962 100644 --- a/tests/components/humidifier/test_init.py +++ b/tests/components/humidifier/test_init.py @@ -13,7 +13,7 @@ ) from homeassistant.core import HomeAssistant -from tests.common import import_and_test_deprecated_constant_enum +from tests.common import help_test_all, import_and_test_deprecated_constant_enum class MockHumidifierEntity(HumidifierEntity): @@ -54,6 +54,15 @@ def _create_tuples(enum: Enum, constant_prefix: str) -> list[tuple[Enum, str]]: return result +@pytest.mark.parametrize( + "module", + [humidifier, humidifier.const], +) +def test_all(module: ModuleType) -> None: + """Test module.__all__ is correctly set.""" + help_test_all(module) + + @pytest.mark.parametrize( ("enum", "constant_prefix"), _create_tuples(humidifier.HumidifierEntityFeature, "SUPPORT_") diff --git a/tests/components/lock/test_init.py b/tests/components/lock/test_init.py index 854b89fd1d8316..7ebb5bf3027b84 100644 --- a/tests/components/lock/test_init.py +++ b/tests/components/lock/test_init.py @@ -28,7 +28,7 @@ from .conftest import MockLock -from tests.common import import_and_test_deprecated_constant_enum +from tests.common import help_test_all, import_and_test_deprecated_constant_enum async def help_test_async_lock_service( @@ -371,6 +371,11 @@ async def test_lock_with_illegal_default_code( assert exc.value.translation_key == "add_default_code" +def test_all() -> None: + """Test module.__all__ is correctly set.""" + help_test_all(lock) + + @pytest.mark.parametrize(("enum"), list(LockEntityFeature)) def test_deprecated_constants( caplog: pytest.LogCaptureFixture, diff --git a/tests/components/number/test_const.py b/tests/components/number/test_const.py index e4b47e17e6edeb..13d94e2eeaf55f 100644 --- a/tests/components/number/test_const.py +++ b/tests/components/number/test_const.py @@ -4,7 +4,12 @@ from homeassistant.components.number import const -from tests.common import import_and_test_deprecated_constant_enum +from tests.common import help_test_all, import_and_test_deprecated_constant_enum + + +def test_all() -> None: + """Test module.__all__ is correctly set.""" + help_test_all(const) @pytest.mark.parametrize(("enum"), list(const.NumberMode)) diff --git a/tests/components/remote/test_init.py b/tests/components/remote/test_init.py index a75ff85848362f..be4a4843097bd1 100644 --- a/tests/components/remote/test_init.py +++ b/tests/components/remote/test_init.py @@ -22,7 +22,11 @@ ) from homeassistant.core import HomeAssistant -from tests.common import async_mock_service, import_and_test_deprecated_constant_enum +from tests.common import ( + async_mock_service, + help_test_all, + import_and_test_deprecated_constant_enum, +) TEST_PLATFORM = {DOMAIN: {CONF_PLATFORM: "test"}} SERVICE_SEND_COMMAND = "send_command" @@ -143,6 +147,11 @@ async def test_delete_command(hass: HomeAssistant) -> None: assert call.data[ATTR_ENTITY_ID] == ENTITY_ID +def test_all() -> None: + """Test module.__all__ is correctly set.""" + help_test_all(remote) + + @pytest.mark.parametrize(("enum"), list(remote.RemoteEntityFeature)) def test_deprecated_constants( caplog: pytest.LogCaptureFixture, diff --git a/tests/components/sensor/test_init.py b/tests/components/sensor/test_init.py index 829bb5af8279e9..522afe3b992e58 100644 --- a/tests/components/sensor/test_init.py +++ b/tests/components/sensor/test_init.py @@ -52,6 +52,7 @@ MockModule, MockPlatform, async_mock_restore_state_shutdown_restart, + help_test_all, import_and_test_deprecated_constant_enum, mock_config_flow, mock_integration, @@ -2524,6 +2525,15 @@ async def test_entity_category_config_raises_error( assert not hass.states.get("sensor.test") +@pytest.mark.parametrize( + "module", + [sensor, sensor.const], +) +def test_all(module: ModuleType) -> None: + """Test module.__all__ is correctly set.""" + help_test_all(module) + + @pytest.mark.parametrize(("enum"), list(sensor.SensorStateClass)) @pytest.mark.parametrize(("module"), [sensor, sensor.const]) def test_deprecated_constants( diff --git a/tests/components/siren/test_init.py b/tests/components/siren/test_init.py index abc5b0fac38b82..1cf44d16ea001e 100644 --- a/tests/components/siren/test_init.py +++ b/tests/components/siren/test_init.py @@ -13,7 +13,7 @@ from homeassistant.components.siren.const import SirenEntityFeature from homeassistant.core import HomeAssistant -from tests.common import import_and_test_deprecated_constant_enum +from tests.common import help_test_all, import_and_test_deprecated_constant_enum class MockSirenEntity(SirenEntity): @@ -110,6 +110,15 @@ async def test_missing_tones_dict(hass: HomeAssistant) -> None: process_turn_on_params(siren, {"tone": 3}) +@pytest.mark.parametrize( + "module", + [siren, siren.const], +) +def test_all(module: ModuleType) -> None: + """Test module.__all__ is correctly set.""" + help_test_all(module) + + @pytest.mark.parametrize(("enum"), list(SirenEntityFeature)) @pytest.mark.parametrize(("module"), [siren, siren.const]) def test_deprecated_constants( diff --git a/tests/components/switch/test_init.py b/tests/components/switch/test_init.py index 7a43e0bf50ef86..deb7acb512a94e 100644 --- a/tests/components/switch/test_init.py +++ b/tests/components/switch/test_init.py @@ -9,7 +9,11 @@ from . import common -from tests.common import MockUser, import_and_test_deprecated_constant_enum +from tests.common import ( + MockUser, + help_test_all, + import_and_test_deprecated_constant_enum, +) @pytest.fixture(autouse=True) @@ -82,6 +86,11 @@ async def test_switch_context( assert state2.context.user_id == hass_admin_user.id +def test_all() -> None: + """Test module.__all__ is correctly set.""" + help_test_all(switch) + + @pytest.mark.parametrize(("enum"), list(switch.SwitchDeviceClass)) def test_deprecated_constants( caplog: pytest.LogCaptureFixture, diff --git a/tests/components/water_heater/test_init.py b/tests/components/water_heater/test_init.py index 861be19234088f..b81ef3694521b2 100644 --- a/tests/components/water_heater/test_init.py +++ b/tests/components/water_heater/test_init.py @@ -17,7 +17,11 @@ from homeassistant.const import UnitOfTemperature from homeassistant.core import HomeAssistant -from tests.common import async_mock_service, import_and_test_deprecated_constant_enum +from tests.common import ( + async_mock_service, + help_test_all, + import_and_test_deprecated_constant_enum, +) async def test_set_temp_schema_no_req( @@ -102,6 +106,11 @@ async def test_sync_turn_off(hass: HomeAssistant) -> None: assert water_heater.async_turn_off.call_count == 1 +def test_all() -> None: + """Test module.__all__ is correctly set.""" + help_test_all(water_heater) + + @pytest.mark.parametrize( ("enum"), [ diff --git a/tests/helpers/test_deprecation.py b/tests/helpers/test_deprecation.py index 017e541bb08e4f..25b37e2073f2a4 100644 --- a/tests/helpers/test_deprecation.py +++ b/tests/helpers/test_deprecation.py @@ -429,7 +429,7 @@ def test_test_check_if_deprecated_constant_invalid( @pytest.mark.parametrize( - ("module_global", "expected"), + ("module_globals", "expected"), [ ({"CONSTANT": 1}, ["CONSTANT"]), ({"_DEPRECATED_CONSTANT": 1}, ["_DEPRECATED_CONSTANT", "CONSTANT"]), @@ -440,7 +440,7 @@ def test_test_check_if_deprecated_constant_invalid( ], ) def test_dir_with_deprecated_constants( - module_global: dict[str, Any], expected: list[str] + module_globals: dict[str, Any], expected: list[str] ) -> None: """Test dir() with deprecated constants.""" - assert dir_with_deprecated_constants(module_global) == expected + assert dir_with_deprecated_constants([*module_globals.keys()]) == expected diff --git a/tests/helpers/test_device_registry.py b/tests/helpers/test_device_registry.py index 43540a52f7daba..240afa2cbab8b2 100644 --- a/tests/helpers/test_device_registry.py +++ b/tests/helpers/test_device_registry.py @@ -20,6 +20,7 @@ from tests.common import ( MockConfigEntry, flush_store, + help_test_all, import_and_test_deprecated_constant_enum, ) @@ -2018,6 +2019,11 @@ async def test_loading_invalid_configuration_url_from_storage( assert entry.configuration_url == "invalid" +def test_all() -> None: + """Test module.__all__ is correctly set.""" + help_test_all(dr) + + @pytest.mark.parametrize(("enum"), list(dr.DeviceEntryDisabler)) def test_deprecated_constants( caplog: pytest.LogCaptureFixture, diff --git a/tests/test_const.py b/tests/test_const.py index fedf35ae6d1b87..4b9be4f27f1f63 100644 --- a/tests/test_const.py +++ b/tests/test_const.py @@ -9,6 +9,7 @@ from homeassistant.components import sensor from tests.common import ( + help_test_all, import_and_test_deprecated_constant, import_and_test_deprecated_constant_enum, ) @@ -23,6 +24,11 @@ def _create_tuples( return result +def test_all() -> None: + """Test module.__all__ is correctly set.""" + help_test_all(const) + + @pytest.mark.parametrize( ("enum", "constant_prefix"), _create_tuples(const.EntityCategory, "ENTITY_CATEGORY_") diff --git a/tests/test_core.py b/tests/test_core.py index 90b87068a5d58f..bbd2715124392c 100644 --- a/tests/test_core.py +++ b/tests/test_core.py @@ -61,6 +61,7 @@ from .common import ( async_capture_events, async_mock_service, + help_test_all, import_and_test_deprecated_constant_enum, ) @@ -2630,6 +2631,11 @@ async def shutdown_func() -> None: assert not evt.is_set() +def test_all() -> None: + """Test module.__all__ is correctly set.""" + help_test_all(ha) + + @pytest.mark.parametrize( ("enum"), [ diff --git a/tests/test_data_entry_flow.py b/tests/test_data_entry_flow.py index eb507febe8a1b8..602b21c15bc40f 100644 --- a/tests/test_data_entry_flow.py +++ b/tests/test_data_entry_flow.py @@ -10,7 +10,11 @@ from homeassistant.core import HomeAssistant from homeassistant.util.decorator import Registry -from .common import async_capture_events, import_and_test_deprecated_constant_enum +from .common import ( + async_capture_events, + help_test_all, + import_and_test_deprecated_constant_enum, +) @pytest.fixture @@ -804,6 +808,11 @@ async def async_step_second(self, user_input=None): assert len(manager.async_progress()) == 0 +def test_all() -> None: + """Test module.__all__ is correctly set.""" + help_test_all(data_entry_flow) + + @pytest.mark.parametrize(("enum"), list(data_entry_flow.FlowResultType)) def test_deprecated_constants( caplog: pytest.LogCaptureFixture, diff --git a/tests/testing_config/custom_components/test_constant_deprecation/__init__.py b/tests/testing_config/custom_components/test_constant_deprecation/__init__.py index 4367cbed7b1af5..b061b9c35fc0be 100644 --- a/tests/testing_config/custom_components/test_constant_deprecation/__init__.py +++ b/tests/testing_config/custom_components/test_constant_deprecation/__init__.py @@ -4,6 +4,6 @@ from typing import Any -def import_deprecated_costant(module: ModuleType, constant_name: str) -> Any: +def import_deprecated_constant(module: ModuleType, constant_name: str) -> Any: """Import and return deprecated constant.""" return getattr(module, constant_name) From 04bf56930891420c0b61997aa20960e7ea5aad3c Mon Sep 17 00:00:00 2001 From: Bram Kragten Date: Thu, 4 Jan 2024 19:45:18 +0100 Subject: [PATCH 16/26] Update frontend to 20240104.0 (#107155) --- homeassistant/components/frontend/manifest.json | 2 +- homeassistant/package_constraints.txt | 2 +- requirements_all.txt | 2 +- requirements_test_all.txt | 2 +- 4 files changed, 4 insertions(+), 4 deletions(-) diff --git a/homeassistant/components/frontend/manifest.json b/homeassistant/components/frontend/manifest.json index 52f3932237b844..ad24f6bb12d79a 100644 --- a/homeassistant/components/frontend/manifest.json +++ b/homeassistant/components/frontend/manifest.json @@ -20,5 +20,5 @@ "documentation": "https://www.home-assistant.io/integrations/frontend", "integration_type": "system", "quality_scale": "internal", - "requirements": ["home-assistant-frontend==20240103.3"] + "requirements": ["home-assistant-frontend==20240104.0"] } diff --git a/homeassistant/package_constraints.txt b/homeassistant/package_constraints.txt index c5715c1c155a2f..655f46a88383a9 100644 --- a/homeassistant/package_constraints.txt +++ b/homeassistant/package_constraints.txt @@ -28,7 +28,7 @@ habluetooth==2.0.2 hass-nabucasa==0.75.1 hassil==1.5.1 home-assistant-bluetooth==1.11.0 -home-assistant-frontend==20240103.3 +home-assistant-frontend==20240104.0 home-assistant-intents==2024.1.2 httpx==0.26.0 ifaddr==0.2.0 diff --git a/requirements_all.txt b/requirements_all.txt index 7f5ef2a7b427f9..0c61bd87ebeb7f 100644 --- a/requirements_all.txt +++ b/requirements_all.txt @@ -1038,7 +1038,7 @@ hole==0.8.0 holidays==0.39 # homeassistant.components.frontend -home-assistant-frontend==20240103.3 +home-assistant-frontend==20240104.0 # homeassistant.components.conversation home-assistant-intents==2024.1.2 diff --git a/requirements_test_all.txt b/requirements_test_all.txt index 0d488cb865436a..6602d21b6a40e2 100644 --- a/requirements_test_all.txt +++ b/requirements_test_all.txt @@ -831,7 +831,7 @@ hole==0.8.0 holidays==0.39 # homeassistant.components.frontend -home-assistant-frontend==20240103.3 +home-assistant-frontend==20240104.0 # homeassistant.components.conversation home-assistant-intents==2024.1.2 From c78d691d30c5cb39aa4729e65c8f673a21d99297 Mon Sep 17 00:00:00 2001 From: Matt Emerick-Law Date: Thu, 4 Jan 2024 19:41:12 +0000 Subject: [PATCH 17/26] Bump Orvibo to 1.1.2 (#107162) * Bump python-orvibo version Fixes https://github.com/home-assistant/core/issues/106923 * Add version number * Remove version * Bump python-orvibo version --- homeassistant/components/orvibo/manifest.json | 2 +- requirements_all.txt | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/homeassistant/components/orvibo/manifest.json b/homeassistant/components/orvibo/manifest.json index 72cdc4118dffa1..05ce5edd8bd313 100644 --- a/homeassistant/components/orvibo/manifest.json +++ b/homeassistant/components/orvibo/manifest.json @@ -5,5 +5,5 @@ "documentation": "https://www.home-assistant.io/integrations/orvibo", "iot_class": "local_push", "loggers": ["orvibo"], - "requirements": ["orvibo==1.1.1"] + "requirements": ["orvibo==1.1.2"] } diff --git a/requirements_all.txt b/requirements_all.txt index 0c61bd87ebeb7f..48658c9dda1c27 100644 --- a/requirements_all.txt +++ b/requirements_all.txt @@ -1443,7 +1443,7 @@ oralb-ble==0.17.6 oru==0.1.11 # homeassistant.components.orvibo -orvibo==1.1.1 +orvibo==1.1.2 # homeassistant.components.ourgroceries ourgroceries==1.5.4 From 3215dfee6db673b752872523ef1552dd06adc143 Mon Sep 17 00:00:00 2001 From: "J. Nick Koston" Date: Thu, 4 Jan 2024 11:21:01 -1000 Subject: [PATCH 18/26] Bump aiohomekit to 3.1.2 (#107177) --- homeassistant/components/homekit_controller/manifest.json | 2 +- requirements_all.txt | 2 +- requirements_test_all.txt | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/homeassistant/components/homekit_controller/manifest.json b/homeassistant/components/homekit_controller/manifest.json index edb81c14a7279e..4af79a6f811b25 100644 --- a/homeassistant/components/homekit_controller/manifest.json +++ b/homeassistant/components/homekit_controller/manifest.json @@ -14,6 +14,6 @@ "documentation": "https://www.home-assistant.io/integrations/homekit_controller", "iot_class": "local_push", "loggers": ["aiohomekit", "commentjson"], - "requirements": ["aiohomekit==3.1.1"], + "requirements": ["aiohomekit==3.1.2"], "zeroconf": ["_hap._tcp.local.", "_hap._udp.local."] } diff --git a/requirements_all.txt b/requirements_all.txt index 48658c9dda1c27..36944f26f18a47 100644 --- a/requirements_all.txt +++ b/requirements_all.txt @@ -257,7 +257,7 @@ aioguardian==2022.07.0 aioharmony==0.2.10 # homeassistant.components.homekit_controller -aiohomekit==3.1.1 +aiohomekit==3.1.2 # homeassistant.components.http aiohttp-fast-url-dispatcher==0.3.0 diff --git a/requirements_test_all.txt b/requirements_test_all.txt index 6602d21b6a40e2..1fa9f2d122ebc6 100644 --- a/requirements_test_all.txt +++ b/requirements_test_all.txt @@ -233,7 +233,7 @@ aioguardian==2022.07.0 aioharmony==0.2.10 # homeassistant.components.homekit_controller -aiohomekit==3.1.1 +aiohomekit==3.1.2 # homeassistant.components.http aiohttp-fast-url-dispatcher==0.3.0 From d7e1a4fa2090c389700e567f2dd909093e62abdb Mon Sep 17 00:00:00 2001 From: Erwin Douna Date: Thu, 4 Jan 2024 23:41:56 +0100 Subject: [PATCH 19/26] Bump to PyTado 0.17.3 (#107181) --- homeassistant/components/tado/manifest.json | 2 +- requirements_all.txt | 2 +- requirements_test_all.txt | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/homeassistant/components/tado/manifest.json b/homeassistant/components/tado/manifest.json index 467697fc8108d7..bae637f31801cd 100644 --- a/homeassistant/components/tado/manifest.json +++ b/homeassistant/components/tado/manifest.json @@ -14,5 +14,5 @@ }, "iot_class": "cloud_polling", "loggers": ["PyTado"], - "requirements": ["python-tado==0.17.0"] + "requirements": ["python-tado==0.17.3"] } diff --git a/requirements_all.txt b/requirements_all.txt index 36944f26f18a47..b3ad3fac35cb1f 100644 --- a/requirements_all.txt +++ b/requirements_all.txt @@ -2241,7 +2241,7 @@ python-smarttub==0.0.36 python-songpal==0.16 # homeassistant.components.tado -python-tado==0.17.0 +python-tado==0.17.3 # homeassistant.components.telegram_bot python-telegram-bot==13.1 diff --git a/requirements_test_all.txt b/requirements_test_all.txt index 1fa9f2d122ebc6..fea1828bd2d98a 100644 --- a/requirements_test_all.txt +++ b/requirements_test_all.txt @@ -1696,7 +1696,7 @@ python-smarttub==0.0.36 python-songpal==0.16 # homeassistant.components.tado -python-tado==0.17.0 +python-tado==0.17.3 # homeassistant.components.telegram_bot python-telegram-bot==13.1 From 4e126d68b79c496beb09feb61727019d97c1cbaa Mon Sep 17 00:00:00 2001 From: Michael <35783820+mib1185@users.noreply.github.com> Date: Thu, 4 Jan 2024 23:36:36 +0100 Subject: [PATCH 20/26] Fix switch states in AVM FRITZ!Box Tools (#107183) --- homeassistant/components/fritz/common.py | 1 + homeassistant/components/fritz/switch.py | 21 ++++++++++----------- 2 files changed, 11 insertions(+), 11 deletions(-) diff --git a/homeassistant/components/fritz/common.py b/homeassistant/components/fritz/common.py index 63f9f593ea8879..bad73d913206bf 100644 --- a/homeassistant/components/fritz/common.py +++ b/homeassistant/components/fritz/common.py @@ -1063,6 +1063,7 @@ class SwitchInfo(TypedDict): type: str callback_update: Callable callback_switch: Callable + init_state: bool class FritzBoxBaseEntity: diff --git a/homeassistant/components/fritz/switch.py b/homeassistant/components/fritz/switch.py index 026c0f3d6fbfe8..c3da6b5af0b8b8 100644 --- a/homeassistant/components/fritz/switch.py +++ b/homeassistant/components/fritz/switch.py @@ -166,9 +166,7 @@ async def _async_wifi_entities_list( _LOGGER.debug("WiFi networks list: %s", networks) return [ - FritzBoxWifiSwitch( - avm_wrapper, device_friendly_name, index, data["switch_name"] - ) + FritzBoxWifiSwitch(avm_wrapper, device_friendly_name, index, data) for index, data in networks.items() ] @@ -310,18 +308,16 @@ async def async_turn_off(self, **kwargs: Any) -> None: await self._async_handle_turn_on_off(turn_on=False) -class FritzBoxBaseSwitch(FritzBoxBaseEntity): +class FritzBoxBaseSwitch(FritzBoxBaseEntity, SwitchEntity): """Fritz switch base class.""" - _attr_is_on: bool | None = False - def __init__( self, avm_wrapper: AvmWrapper, device_friendly_name: str, switch_info: SwitchInfo, ) -> None: - """Init Fritzbox port switch.""" + """Init Fritzbox base switch.""" super().__init__(avm_wrapper, device_friendly_name) self._description = switch_info["description"] @@ -330,6 +326,7 @@ def __init__( self._type = switch_info["type"] self._update = switch_info["callback_update"] self._switch = switch_info["callback_switch"] + self._attr_is_on = switch_info["init_state"] self._name = f"{self._friendly_name} {self._description}" self._unique_id = f"{self._avm_wrapper.unique_id}-{slugify(self._description)}" @@ -381,7 +378,7 @@ async def _async_handle_turn_on_off(self, turn_on: bool) -> None: self._attr_is_on = turn_on -class FritzBoxPortSwitch(FritzBoxBaseSwitch, SwitchEntity): +class FritzBoxPortSwitch(FritzBoxBaseSwitch): """Defines a FRITZ!Box Tools PortForward switch.""" def __init__( @@ -412,6 +409,7 @@ def __init__( type=SWITCH_TYPE_PORTFORWARD, callback_update=self._async_fetch_update, callback_switch=self._async_switch_on_off_executor, + init_state=port_mapping["NewEnabled"], ) super().__init__(avm_wrapper, device_friendly_name, switch_info) @@ -553,7 +551,7 @@ async def _async_handle_turn_on_off(self, turn_on: bool) -> bool: return True -class FritzBoxWifiSwitch(FritzBoxBaseSwitch, SwitchEntity): +class FritzBoxWifiSwitch(FritzBoxBaseSwitch): """Defines a FRITZ!Box Tools Wifi switch.""" def __init__( @@ -561,7 +559,7 @@ def __init__( avm_wrapper: AvmWrapper, device_friendly_name: str, network_num: int, - network_name: str, + network_data: dict, ) -> None: """Init Fritz Wifi switch.""" self._avm_wrapper = avm_wrapper @@ -571,12 +569,13 @@ def __init__( self._network_num = network_num switch_info = SwitchInfo( - description=f"Wi-Fi {network_name}", + description=f"Wi-Fi {network_data['switch_name']}", friendly_name=device_friendly_name, icon="mdi:wifi", type=SWITCH_TYPE_WIFINETWORK, callback_update=self._async_fetch_update, callback_switch=self._async_switch_on_off_executor, + init_state=network_data["enabled"], ) super().__init__(self._avm_wrapper, device_friendly_name, switch_info) From c242dcd1f20b48db9e3734bbab7d5374e3216828 Mon Sep 17 00:00:00 2001 From: Brett Adams Date: Fri, 5 Jan 2024 17:23:43 +1000 Subject: [PATCH 21/26] Hotfix cache logic bug in Tessie (#107187) --- homeassistant/components/tessie/coordinator.py | 4 +--- 1 file changed, 1 insertion(+), 3 deletions(-) diff --git a/homeassistant/components/tessie/coordinator.py b/homeassistant/components/tessie/coordinator.py index c2f53da53bc30d..75cac088bde47c 100644 --- a/homeassistant/components/tessie/coordinator.py +++ b/homeassistant/components/tessie/coordinator.py @@ -41,7 +41,6 @@ def __init__( self.vin = vin self.session = async_get_clientsession(hass) self.data = self._flatten(data) - self.did_first_update = False async def _async_update_data(self) -> dict[str, Any]: """Update vehicle data using Tessie API.""" @@ -50,7 +49,7 @@ async def _async_update_data(self) -> dict[str, Any]: session=self.session, api_key=self.api_key, vin=self.vin, - use_cache=self.did_first_update, + use_cache=False, ) except ClientResponseError as e: if e.status == HTTPStatus.UNAUTHORIZED: @@ -58,7 +57,6 @@ async def _async_update_data(self) -> dict[str, Any]: raise ConfigEntryAuthFailed from e raise e - self.did_first_update = True if vehicle["state"] == TessieStatus.ONLINE: # Vehicle is online, all data is fresh return self._flatten(vehicle) From 4ade5e46d9e3c4b5a8a3b3d88b40286cd1ae304b Mon Sep 17 00:00:00 2001 From: tronikos Date: Thu, 4 Jan 2024 17:07:15 -0800 Subject: [PATCH 22/26] Disable IPv6 in the opower integration to fix AEP utilities (#107203) --- homeassistant/components/opower/config_flow.py | 3 ++- homeassistant/components/opower/coordinator.py | 3 ++- 2 files changed, 4 insertions(+), 2 deletions(-) diff --git a/homeassistant/components/opower/config_flow.py b/homeassistant/components/opower/config_flow.py index d456fc536e58ff..ab1fbbe36e3b5d 100644 --- a/homeassistant/components/opower/config_flow.py +++ b/homeassistant/components/opower/config_flow.py @@ -3,6 +3,7 @@ from collections.abc import Mapping import logging +import socket from typing import Any from opower import ( @@ -38,7 +39,7 @@ async def _validate_login( ) -> dict[str, str]: """Validate login data and return any errors.""" api = Opower( - async_create_clientsession(hass), + async_create_clientsession(hass, family=socket.AF_INET), login_data[CONF_UTILITY], login_data[CONF_USERNAME], login_data[CONF_PASSWORD], diff --git a/homeassistant/components/opower/coordinator.py b/homeassistant/components/opower/coordinator.py index a474255e34dce7..73c60068cd4571 100644 --- a/homeassistant/components/opower/coordinator.py +++ b/homeassistant/components/opower/coordinator.py @@ -1,6 +1,7 @@ """Coordinator to handle Opower connections.""" from datetime import datetime, timedelta import logging +import socket from types import MappingProxyType from typing import Any, cast @@ -51,7 +52,7 @@ def __init__( update_interval=timedelta(hours=12), ) self.api = Opower( - aiohttp_client.async_get_clientsession(hass), + aiohttp_client.async_get_clientsession(hass, family=socket.AF_INET), entry_data[CONF_UTILITY], entry_data[CONF_USERNAME], entry_data[CONF_PASSWORD], From c3963b26e70ef795861ce89baef9a92208fb72b7 Mon Sep 17 00:00:00 2001 From: Petru Paler Date: Fri, 5 Jan 2024 08:18:25 +0000 Subject: [PATCH 23/26] Fix entity property cache creation arguments (#107221) --- homeassistant/helpers/entity.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/homeassistant/helpers/entity.py b/homeassistant/helpers/entity.py index 3c3c8474e67a26..743d3675a3b1c4 100644 --- a/homeassistant/helpers/entity.py +++ b/homeassistant/helpers/entity.py @@ -292,7 +292,7 @@ def __new__( Pop cached_properties and store it in the namespace. """ namespace["_CachedProperties__cached_properties"] = cached_properties or set() - return super().__new__(mcs, name, bases, namespace) + return super().__new__(mcs, name, bases, namespace, **kwargs) def __init__( cls, From 056701d21822de717fee8ecc48184600cf494cc0 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Joakim=20S=C3=B8rensen?= Date: Fri, 5 Jan 2024 10:38:54 +0100 Subject: [PATCH 24/26] Use supported_features_compat in update.install service (#107224) --- homeassistant/components/update/__init__.py | 4 +- tests/components/update/test_init.py | 72 +++++++++++++++++++++ 2 files changed, 74 insertions(+), 2 deletions(-) diff --git a/homeassistant/components/update/__init__.py b/homeassistant/components/update/__init__.py index 40431332aafedf..8ec14b6e3a89f1 100644 --- a/homeassistant/components/update/__init__.py +++ b/homeassistant/components/update/__init__.py @@ -140,7 +140,7 @@ async def async_install(entity: UpdateEntity, service_call: ServiceCall) -> None # If version is specified, but not supported by the entity. if ( version is not None - and UpdateEntityFeature.SPECIFIC_VERSION not in entity.supported_features + and UpdateEntityFeature.SPECIFIC_VERSION not in entity.supported_features_compat ): raise HomeAssistantError( f"Installing a specific version is not supported for {entity.entity_id}" @@ -149,7 +149,7 @@ async def async_install(entity: UpdateEntity, service_call: ServiceCall) -> None # If backup is requested, but not supported by the entity. if ( backup := service_call.data[ATTR_BACKUP] - ) and UpdateEntityFeature.BACKUP not in entity.supported_features: + ) and UpdateEntityFeature.BACKUP not in entity.supported_features_compat: raise HomeAssistantError(f"Backup is not supported for {entity.entity_id}") # Update is already in progress. diff --git a/tests/components/update/test_init.py b/tests/components/update/test_init.py index 92e63af4b6f9a7..67661d6936efa6 100644 --- a/tests/components/update/test_init.py +++ b/tests/components/update/test_init.py @@ -885,3 +885,75 @@ def supported_features(self) -> int: caplog.clear() assert entity.supported_features_compat is UpdateEntityFeature(1) assert "is using deprecated supported features values" not in caplog.text + + +async def test_deprecated_supported_features_ints_with_service_call( + hass: HomeAssistant, + caplog: pytest.LogCaptureFixture, +) -> None: + """Test deprecated supported features ints with install service.""" + + async def async_setup_entry_init( + hass: HomeAssistant, config_entry: ConfigEntry + ) -> bool: + """Set up test config entry.""" + await hass.config_entries.async_forward_entry_setup(config_entry, DOMAIN) + return True + + mock_platform(hass, f"{TEST_DOMAIN}.config_flow") + mock_integration( + hass, + MockModule( + TEST_DOMAIN, + async_setup_entry=async_setup_entry_init, + ), + ) + + class MockUpdateEntity(UpdateEntity): + _attr_supported_features = 1 | 2 + + def install(self, version: str | None = None, backup: bool = False) -> None: + """Install an update.""" + + entity = MockUpdateEntity() + entity.entity_id = ( + "update.test_deprecated_supported_features_ints_with_service_call" + ) + + async def async_setup_entry_platform( + hass: HomeAssistant, + config_entry: ConfigEntry, + async_add_entities: AddEntitiesCallback, + ) -> None: + """Set up test update platform via config entry.""" + async_add_entities([entity]) + + mock_platform( + hass, + f"{TEST_DOMAIN}.{DOMAIN}", + MockPlatform(async_setup_entry=async_setup_entry_platform), + ) + + config_entry = MockConfigEntry(domain=TEST_DOMAIN) + config_entry.add_to_hass(hass) + assert await hass.config_entries.async_setup(config_entry.entry_id) + await hass.async_block_till_done() + + assert "is using deprecated supported features values" in caplog.text + + assert isinstance(entity.supported_features, int) + + with pytest.raises( + HomeAssistantError, + match="Backup is not supported for update.test_deprecated_supported_features_ints_with_service_call", + ): + await hass.services.async_call( + DOMAIN, + SERVICE_INSTALL, + { + ATTR_VERSION: "0.9.9", + ATTR_BACKUP: True, + ATTR_ENTITY_ID: "update.test_deprecated_supported_features_ints_with_service_call", + }, + blocking=True, + ) From d012817190520c43890616c4a96701fcc1158920 Mon Sep 17 00:00:00 2001 From: Raman Gupta <7243222+raman325@users.noreply.github.com> Date: Fri, 5 Jan 2024 06:38:00 -0500 Subject: [PATCH 25/26] Bump zwave-js-server-python to 0.55.3 (#107225) Co-authored-by: Martin Hjelmare --- homeassistant/components/zwave_js/manifest.json | 2 +- requirements_all.txt | 2 +- requirements_test_all.txt | 2 +- tests/components/zwave_js/test_events.py | 11 ++++++++--- 4 files changed, 11 insertions(+), 6 deletions(-) diff --git a/homeassistant/components/zwave_js/manifest.json b/homeassistant/components/zwave_js/manifest.json index 9a66dae8e9323b..a06de5cb8eeba2 100644 --- a/homeassistant/components/zwave_js/manifest.json +++ b/homeassistant/components/zwave_js/manifest.json @@ -9,7 +9,7 @@ "iot_class": "local_push", "loggers": ["zwave_js_server"], "quality_scale": "platinum", - "requirements": ["pyserial==3.5", "zwave-js-server-python==0.55.2"], + "requirements": ["pyserial==3.5", "zwave-js-server-python==0.55.3"], "usb": [ { "vid": "0658", diff --git a/requirements_all.txt b/requirements_all.txt index b3ad3fac35cb1f..012c601b785189 100644 --- a/requirements_all.txt +++ b/requirements_all.txt @@ -2893,7 +2893,7 @@ zigpy==0.60.4 zm-py==0.5.2 # homeassistant.components.zwave_js -zwave-js-server-python==0.55.2 +zwave-js-server-python==0.55.3 # homeassistant.components.zwave_me zwave-me-ws==0.4.3 diff --git a/requirements_test_all.txt b/requirements_test_all.txt index fea1828bd2d98a..96775e057914e0 100644 --- a/requirements_test_all.txt +++ b/requirements_test_all.txt @@ -2189,7 +2189,7 @@ zigpy-znp==0.12.1 zigpy==0.60.4 # homeassistant.components.zwave_js -zwave-js-server-python==0.55.2 +zwave-js-server-python==0.55.3 # homeassistant.components.zwave_me zwave-me-ws==0.4.3 diff --git a/tests/components/zwave_js/test_events.py b/tests/components/zwave_js/test_events.py index 4fbaa97f118e02..1e91b9338fad65 100644 --- a/tests/components/zwave_js/test_events.py +++ b/tests/components/zwave_js/test_events.py @@ -348,7 +348,11 @@ async def test_power_level_notification( async def test_unknown_notification( - hass: HomeAssistant, hank_binary_switch, integration, client + hass: HomeAssistant, + caplog: pytest.LogCaptureFixture, + hank_binary_switch, + integration, + client, ) -> None: """Test behavior of unknown notification type events.""" # just pick a random node to fake the notification event @@ -358,8 +362,9 @@ async def test_unknown_notification( # by the lib. We will use a class that is guaranteed not to be recognized notification_obj = AsyncMock() notification_obj.node = node - with pytest.raises(TypeError): - node.emit("notification", {"notification": notification_obj}) + node.emit("notification", {"notification": notification_obj}) + + assert f"Unhandled notification type: {notification_obj}" in caplog.text notification_events = async_capture_events(hass, "zwave_js_notification") From 658f1cf5c5e83946e9ca394542ccbc70de212344 Mon Sep 17 00:00:00 2001 From: Franck Nijhof Date: Fri, 5 Jan 2024 13:01:35 +0100 Subject: [PATCH 26/26] Bump version to 2024.1.1 --- homeassistant/const.py | 2 +- pyproject.toml | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/homeassistant/const.py b/homeassistant/const.py index 9e0505fadf333f..cea73ec243b47b 100644 --- a/homeassistant/const.py +++ b/homeassistant/const.py @@ -16,7 +16,7 @@ APPLICATION_NAME: Final = "HomeAssistant" MAJOR_VERSION: Final = 2024 MINOR_VERSION: Final = 1 -PATCH_VERSION: Final = "0" +PATCH_VERSION: Final = "1" __short_version__: Final = f"{MAJOR_VERSION}.{MINOR_VERSION}" __version__: Final = f"{__short_version__}.{PATCH_VERSION}" REQUIRED_PYTHON_VER: Final[tuple[int, int, int]] = (3, 11, 0) diff --git a/pyproject.toml b/pyproject.toml index 2fc5f594157b0a..3bec11ced3bbe2 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -4,7 +4,7 @@ build-backend = "setuptools.build_meta" [project] name = "homeassistant" -version = "2024.1.0" +version = "2024.1.1" license = {text = "Apache-2.0"} description = "Open-source home automation platform running on Python 3." readme = "README.rst"