**Set this to `False` if your Device Tracker has a GPS Accuracy (`gps_accuracy`) attribute, but it always shows 0 even if the latitude and longitude are correct.**
+`Use GPS Accuracy` | `No` | `True` | Use GPS Accuracy when determining whether to update the places sensor (if 0, don't update the places sensor). By not updating when GPS Accuracy is 0, should prevent inaccurate locations from being set in the places sensors.
**Set this to `False` if your Device Tracker has a GPS Accuracy (`gps_accuracy`) attribute, but it always shows 0 even if the latitude and longitude are correct.**
Advanced Display Options
diff --git a/custom_components/places/__init__.py b/custom_components/places/__init__.py
index 355a7801..f62cd986 100644
--- a/custom_components/places/__init__.py
+++ b/custom_components/places/__init__.py
@@ -1,24 +1,27 @@
+"""Initialize Home Assistant places integration."""
+
+from collections.abc import Callable, MutableMapping
import logging
+from typing import Any
-from homeassistant import config_entries, core
+from homeassistant.config_entries import ConfigEntry
from homeassistant.const import Platform
+from homeassistant.core import HomeAssistant
from homeassistant.helpers import config_validation as cv
from .const import DOMAIN
-_LOGGER = logging.getLogger(__name__)
+_LOGGER: logging.Logger = logging.getLogger(__name__)
PLATFORMS: list[str] = [Platform.SENSOR]
-CONFIG_SCHEMA = cv.empty_config_schema(DOMAIN)
+CONFIG_SCHEMA: Callable[[dict], dict] = cv.empty_config_schema(DOMAIN)
-async def async_setup_entry(
- hass: core.HomeAssistant, entry: config_entries.ConfigEntry
-) -> bool:
+async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
"""Set up from a config entry."""
# _LOGGER.debug(f"[init async_setup_entry] entry: {entry.data}")
hass.data.setdefault(DOMAIN, {})
- hass_data = dict(entry.data)
+ hass_data: MutableMapping[str, Any] = dict(entry.data)
hass.data[DOMAIN][entry.entry_id] = hass_data
# This creates each HA object for each platform your device requires.
@@ -26,15 +29,13 @@ async def async_setup_entry(
return True
-async def async_unload_entry(
- hass: core.HomeAssistant, entry: config_entries.ConfigEntry
-) -> bool:
+async def async_unload_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
"""Unload a config entry."""
# This is called when an entry/configured device is to be removed. The class
# needs to unload itself, and remove callbacks. See the classes for further
# details
- _LOGGER.info(f"Unloading: {entry.data}")
- unload_ok = await hass.config_entries.async_unload_platforms(entry, PLATFORMS)
+ _LOGGER.info("Unloading: %s", entry.data)
+ unload_ok: bool = await hass.config_entries.async_unload_platforms(entry, PLATFORMS)
if unload_ok:
hass.data[DOMAIN].pop(entry.entry_id)
diff --git a/custom_components/places/config_flow.py b/custom_components/places/config_flow.py
index 40239877..b6ed7ebb 100644
--- a/custom_components/places/config_flow.py
+++ b/custom_components/places/config_flow.py
@@ -1,10 +1,14 @@
+"""Config Flow for places integration."""
+
from __future__ import annotations
+from collections.abc import MutableMapping
import logging
from typing import Any
import voluptuous as vol
-from homeassistant import config_entries, core
+
+from homeassistant.config_entries import ConfigEntry, ConfigFlow, ConfigFlowResult, OptionsFlow
from homeassistant.const import (
ATTR_FRIENDLY_NAME,
CONF_API_KEY,
@@ -12,7 +16,7 @@
CONF_LONGITUDE,
CONF_NAME,
)
-from homeassistant.data_entry_flow import FlowResult
+from homeassistant.core import HomeAssistant, callback
from homeassistant.helpers import selector
from .const import (
@@ -40,15 +44,13 @@
TRACKING_DOMAINS_NEED_LATLONG,
)
-_LOGGER = logging.getLogger(__name__)
-MAP_PROVIDER_OPTIONS = ["apple", "google", "osm"]
-STATE_OPTIONS = ["zone, place", "formatted_place", "zone_name, place"]
-DATE_FORMAT_OPTIONS = ["mm/dd", "dd/mm"]
-MAP_ZOOM_MIN = 1
-MAP_ZOOM_MAX = 20
-COMPONENT_CONFIG_URL = (
- "https://github.com/custom-components/places#configuration-options"
-)
+_LOGGER: logging.Logger = logging.getLogger(__name__)
+MAP_PROVIDER_OPTIONS: list[str] = ["apple", "google", "osm"]
+STATE_OPTIONS: list[str] = ["zone, place", "formatted_place", "zone_name, place"]
+DATE_FORMAT_OPTIONS: list[str] = ["mm/dd", "dd/mm"]
+MAP_ZOOM_MIN: int = 1
+MAP_ZOOM_MAX: int = 20
+COMPONENT_CONFIG_URL: str = "https://github.com/custom-components/places#configuration-options"
# Note the input displayed to the user will be translated. See the
# translations/.json file and strings.json. See here for further information:
@@ -56,10 +58,10 @@
def get_devicetracker_id_entities(
- hass: core.HomeAssistant, current_entity=None
-) -> list[str]:
+ hass: HomeAssistant, current_entity=None
+) -> list[selector.SelectOptionDict]:
"""Get the list of valid entities. For sensors, only include ones with latitude and longitude attributes."""
- dt_list = []
+ dt_list: list[selector.SelectOptionDict] = []
for dom in TRACKING_DOMAINS:
# _LOGGER.debug(f"Geting entities for domain: {dom}")
for ent in hass.states.async_all(dom):
@@ -68,26 +70,24 @@ def get_devicetracker_id_entities(
and CONF_LONGITUDE in hass.states.get(ent.entity_id).attributes
):
# _LOGGER.debug(f"Entity: {ent}")
- dt_list.append(
- selector.SelectOptionDict(
- value=str(ent.entity_id),
- label=f"{ent.attributes.get(ATTR_FRIENDLY_NAME)} ({ent.entity_id})",
- )
+ dt_list.extend(
+ [
+ selector.SelectOptionDict(
+ value=str(ent.entity_id),
+ label=f"{ent.attributes.get(ATTR_FRIENDLY_NAME)} ({ent.entity_id})",
+ )
+ ]
)
# Optional: Include the current entity in the list as well.
if current_entity is not None:
# _LOGGER.debug(f"current_entity: {current_entity}")
- dt_list_entities = [d["value"] for d in dt_list]
- if (
- current_entity not in dt_list_entities
- and hass.states.get(current_entity) is not None
- ):
+ dt_list_entities: list[str] = [d["value"] for d in dt_list]
+ if current_entity not in dt_list_entities and hass.states.get(current_entity) is not None:
if (
ATTR_FRIENDLY_NAME in hass.states.get(current_entity).attributes
- and hass.states.get(current_entity).attributes.get(ATTR_FRIENDLY_NAME)
- is not None
+ and hass.states.get(current_entity).attributes.get(ATTR_FRIENDLY_NAME) is not None
):
- current_name = hass.states.get(current_entity).attributes.get(
+ current_name: str = hass.states.get(current_entity).attributes.get(
ATTR_FRIENDLY_NAME
)
# _LOGGER.debug(f"current_name: {current_name}")
@@ -105,7 +105,9 @@ def get_devicetracker_id_entities(
)
)
if dt_list:
- dt_list_sorted = sorted(dt_list, key=lambda d: d["label"].casefold())
+ dt_list_sorted: list[selector.SelectOptionDict] = sorted(
+ dt_list, key=lambda d: d["label"].casefold()
+ )
else:
dt_list_sorted = []
@@ -113,28 +115,34 @@ def get_devicetracker_id_entities(
return dt_list_sorted
-def get_home_zone_entities(hass: core.HomeAssistant) -> list[str]:
+def get_home_zone_entities(hass: HomeAssistant) -> list[selector.SelectOptionDict]:
"""Get the list of valid zones."""
- zone_list = []
+ zone_list: list[selector.SelectOptionDict] = []
for dom in HOME_LOCATION_DOMAINS:
# _LOGGER.debug(f"Geting entities for domain: {dom}")
for ent in hass.states.async_all(dom):
# _LOGGER.debug(f"Entity: {ent}")
- zone_list.append(
- selector.SelectOptionDict(
- value=str(ent.entity_id),
- label=f"{ent.attributes.get(ATTR_FRIENDLY_NAME)} ({ent.entity_id})",
- )
+ zone_list.extend(
+ [
+ selector.SelectOptionDict(
+ value=str(ent.entity_id),
+ label=f"{ent.attributes.get(ATTR_FRIENDLY_NAME)} ({ent.entity_id})",
+ )
+ ]
)
if zone_list:
- zone_list_sorted = sorted(zone_list, key=lambda d: d["label"].casefold())
+ zone_list_sorted: list[selector.SelectOptionDict] = sorted(
+ zone_list, key=lambda d: d["label"].casefold()
+ )
else:
zone_list_sorted = []
# _LOGGER.debug(f"Zones: {zone_list_sorted}")
return zone_list_sorted
-async def validate_input(hass: core.HomeAssistant, data: dict) -> dict[str, Any]:
+async def validate_input(
+ _: HomeAssistant, data: MutableMapping[str, Any]
+) -> MutableMapping[str, Any]:
"""Validate the user input allows us to connect.
Data has the keys from DATA_SCHEMA with values provided by the user.
@@ -145,11 +153,15 @@ async def validate_input(hass: core.HomeAssistant, data: dict) -> dict[str, Any]
return {"title": data[CONF_NAME]}
-class PlacesConfigFlow(config_entries.ConfigFlow, domain=DOMAIN):
+class PlacesConfigFlow(ConfigFlow, domain=DOMAIN):
+ """Config Flow for places integration."""
+
VERSION = 1
- # Connection classes in homeassistant/config_entries.py are now deprecated
+ # Connection classes in homeassistant/py are now deprecated
- async def async_step_user(self, user_input=None) -> FlowResult:
+ async def async_step_user(
+ self, user_input: MutableMapping[str, Any] | None = None
+ ) -> ConfigFlowResult:
"""Handle the initial step."""
# This goes through the steps to take the user through the setup process.
# Using this it is possible to update the UI and prompt for additional
@@ -157,22 +169,22 @@ async def async_step_user(self, user_input=None) -> FlowResult:
# and when that has some validated input, it calls `async_create_entry` to
# actually create the HA config entry. Note the "title" value is returned by
# `validate_input` above.
- errors = {}
+ errors: MutableMapping[str, Any] = {}
if user_input is not None:
try:
- info = await validate_input(self.hass, user_input)
+ info: MutableMapping[str, Any] = await validate_input(self.hass, user_input)
# _LOGGER.debug(f"[New Sensor] info: {info}")
- _LOGGER.debug(f"[New Sensor] user_input: {user_input}")
+ _LOGGER.debug("[New Sensor] user_input: %s", user_input)
return self.async_create_entry(title=info["title"], data=user_input)
- except Exception as err:
- _LOGGER.exception(
- f"[config_flow async_step_user] Unexpected exception: {err}"
- )
+ except Exception as err: # noqa: BLE001
+ _LOGGER.error("[config_flow async_step_user] Unexpected exception: %s", err)
errors["base"] = "unknown"
- devicetracker_id_list = get_devicetracker_id_entities(self.hass)
+ devicetracker_id_list: list[selector.SelectOptionDict] = get_devicetracker_id_entities(
+ self.hass
+ )
zone_list = get_home_zone_entities(self.hass)
# _LOGGER.debug(f"Trackable entities with lat/long: {devicetracker_id_list}")
- DATA_SCHEMA = vol.Schema(
+ DATA_SCHEMA: vol.Schema = vol.Schema(
{
vol.Required(CONF_NAME): str,
vol.Required(CONF_DEVICETRACKER_ID): selector.SelectSelector(
@@ -194,9 +206,7 @@ async def async_step_user(self, user_input=None) -> FlowResult:
mode=selector.SelectSelectorMode.DROPDOWN,
)
),
- vol.Optional(
- CONF_HOME_ZONE, default=DEFAULT_HOME_ZONE
- ): selector.SelectSelector(
+ vol.Optional(CONF_HOME_ZONE, default=DEFAULT_HOME_ZONE): selector.SelectSelector(
selector.SelectSelectorConfig(
options=zone_list,
multiple=False,
@@ -214,9 +224,7 @@ async def async_step_user(self, user_input=None) -> FlowResult:
mode=selector.SelectSelectorMode.DROPDOWN,
)
),
- vol.Optional(
- CONF_MAP_ZOOM, default=int(DEFAULT_MAP_ZOOM)
- ): selector.NumberSelector(
+ vol.Optional(CONF_MAP_ZOOM, default=int(DEFAULT_MAP_ZOOM)): selector.NumberSelector(
selector.NumberSelectorConfig(
min=MAP_ZOOM_MIN,
max=MAP_ZOOM_MAX,
@@ -227,9 +235,9 @@ async def async_step_user(self, user_input=None) -> FlowResult:
vol.Optional(
CONF_EXTENDED_ATTR, default=DEFAULT_EXTENDED_ATTR
): selector.BooleanSelector(selector.BooleanSelectorConfig()),
- vol.Optional(
- CONF_SHOW_TIME, default=DEFAULT_SHOW_TIME
- ): selector.BooleanSelector(selector.BooleanSelectorConfig()),
+ vol.Optional(CONF_SHOW_TIME, default=DEFAULT_SHOW_TIME): selector.BooleanSelector(
+ selector.BooleanSelectorConfig()
+ ),
vol.Optional(
CONF_DATE_FORMAT, default=DEFAULT_DATE_FORMAT
): selector.SelectSelector(
@@ -240,9 +248,9 @@ async def async_step_user(self, user_input=None) -> FlowResult:
mode=selector.SelectSelectorMode.DROPDOWN,
)
),
- vol.Optional(
- CONF_USE_GPS, default=DEFAULT_USE_GPS
- ): selector.BooleanSelector(selector.BooleanSelectorConfig()),
+ vol.Optional(CONF_USE_GPS, default=DEFAULT_USE_GPS): selector.BooleanSelector(
+ selector.BooleanSelectorConfig()
+ ),
}
)
# If there is no user input or there were errors, show the form again, including any errors that were found with the input.
@@ -256,59 +264,54 @@ async def async_step_user(self, user_input=None) -> FlowResult:
)
@staticmethod
- @core.callback
+ @callback
def async_get_options_flow(
- config_entry: config_entries.ConfigEntry,
+ config_entry: ConfigEntry,
) -> PlacesOptionsFlowHandler:
"""Options callback for Places."""
return PlacesOptionsFlowHandler(config_entry)
-class PlacesOptionsFlowHandler(config_entries.OptionsFlow):
+class PlacesOptionsFlowHandler(OptionsFlow):
"""Config flow options for Places. Does not actually store these into Options but updates the Config instead."""
- def __init__(self, entry: config_entries.ConfigEntry) -> None:
+ def __init__(self, entry: ConfigEntry) -> None:
"""Initialize Places options flow."""
self.config_entry = entry
- async def async_step_init(self, user_input=None):
+ async def async_step_init(
+ self, user_input: MutableMapping[str, Any] | None = None
+ ) -> ConfigFlowResult:
"""Manage the options."""
if user_input is not None:
# _LOGGER.debug(f"[options_flow async_step_init] user_input initial: {user_input}")
# Bring in other keys not in the Options Flow
- for m in dict(self.config_entry.data).keys():
+ for m in dict(self.config_entry.data):
user_input.setdefault(m, self.config_entry.data[m])
# Remove any keys with blank values
- for m in dict(user_input).keys():
+ for m in dict(user_input):
# _LOGGER.debug(f"[Options Update] {m} [{type(user_input.get(m))}]: {user_input.get(m)}")
if isinstance(user_input.get(m), str) and not user_input.get(m):
user_input.pop(m)
# _LOGGER.debug(f"[Options Update] updated config: {user_input}")
- self.hass.config_entries.async_update_entry(
+ self.hass.async_update_entry(
self.config_entry, data=user_input, options=self.config_entry.options
)
- await self.hass.config_entries.async_reload(self.config_entry.entry_id)
+ await self.hass.async_reload(self.config_entry.entry_id)
return self.async_create_entry(title="", data={})
# Include the current entity in the list as well. Although it may still fail in validation checking.
- devicetracker_id_list = get_devicetracker_id_entities(
- self.hass,
- self.config_entry.data[CONF_DEVICETRACKER_ID]
- if CONF_DEVICETRACKER_ID in self.config_entry.data
- else None,
+ devicetracker_id_list: list[selector.SelectOptionDict] = get_devicetracker_id_entities(
+ self.hass, self.config_entry.data.get(CONF_DEVICETRACKER_ID, None)
)
- zone_list = get_home_zone_entities(self.hass)
+ zone_list: list[selector.SelectOptionDict] = get_home_zone_entities(self.hass)
# _LOGGER.debug(f"Trackable entities including sensors with lat/long: {devicetracker_id_list}")
- OPTIONS_SCHEMA = vol.Schema(
+ OPTIONS_SCHEMA: vol.Schema = vol.Schema(
{
vol.Required(
CONF_DEVICETRACKER_ID,
- default=(
- self.config_entry.data[CONF_DEVICETRACKER_ID]
- if CONF_DEVICETRACKER_ID in self.config_entry.data
- else None
- ),
+ default=(self.config_entry.data.get(CONF_DEVICETRACKER_ID, None)),
): selector.SelectSelector(
selector.SelectSelectorConfig(
options=devicetracker_id_list,
@@ -320,19 +323,15 @@ async def async_step_init(self, user_input=None):
vol.Optional(
CONF_API_KEY,
default="",
- description={
- "suggested_value": self.config_entry.data[CONF_API_KEY]
- if CONF_API_KEY in self.config_entry.data
- else None
- },
+ description={"suggested_value": self.config_entry.data.get(CONF_API_KEY, None)},
): str,
vol.Optional(
CONF_DISPLAY_OPTIONS,
default=DEFAULT_DISPLAY_OPTIONS,
description={
- "suggested_value": self.config_entry.data[CONF_DISPLAY_OPTIONS]
- if CONF_DISPLAY_OPTIONS in self.config_entry.data
- else DEFAULT_DISPLAY_OPTIONS
+ "suggested_value": self.config_entry.data.get(
+ CONF_DISPLAY_OPTIONS, DEFAULT_DISPLAY_OPTIONS
+ )
},
): selector.SelectSelector(
selector.SelectSelectorConfig(
@@ -346,9 +345,7 @@ async def async_step_init(self, user_input=None):
CONF_HOME_ZONE,
default="",
description={
- "suggested_value": self.config_entry.data[CONF_HOME_ZONE]
- if CONF_HOME_ZONE in self.config_entry.data
- else None
+ "suggested_value": self.config_entry.data.get(CONF_HOME_ZONE, None)
},
): selector.SelectSelector(
selector.SelectSelectorConfig(
@@ -362,9 +359,9 @@ async def async_step_init(self, user_input=None):
CONF_MAP_PROVIDER,
default=DEFAULT_MAP_PROVIDER,
description={
- "suggested_value": self.config_entry.data[CONF_MAP_PROVIDER]
- if CONF_MAP_PROVIDER in self.config_entry.data
- else DEFAULT_MAP_PROVIDER
+ "suggested_value": self.config_entry.data.get(
+ CONF_MAP_PROVIDER, DEFAULT_MAP_PROVIDER
+ )
},
): selector.SelectSelector(
selector.SelectSelectorConfig(
@@ -378,9 +375,9 @@ async def async_step_init(self, user_input=None):
CONF_MAP_ZOOM,
default=DEFAULT_MAP_ZOOM,
description={
- "suggested_value": self.config_entry.data[CONF_MAP_ZOOM]
- if CONF_MAP_ZOOM in self.config_entry.data
- else DEFAULT_MAP_ZOOM
+ "suggested_value": self.config_entry.data.get(
+ CONF_MAP_ZOOM, DEFAULT_MAP_ZOOM
+ )
},
): selector.NumberSelector(
selector.NumberSelectorConfig(
@@ -393,34 +390,24 @@ async def async_step_init(self, user_input=None):
CONF_LANGUAGE,
default="",
description={
- "suggested_value": self.config_entry.data[CONF_LANGUAGE]
- if CONF_LANGUAGE in self.config_entry.data
- else None
+ "suggested_value": self.config_entry.data.get(CONF_LANGUAGE, None)
},
): str,
vol.Optional(
CONF_EXTENDED_ATTR,
- default=(
- self.config_entry.data[CONF_EXTENDED_ATTR]
- if CONF_EXTENDED_ATTR in self.config_entry.data
- else DEFAULT_EXTENDED_ATTR
- ),
+ default=(self.config_entry.data.get(CONF_EXTENDED_ATTR, DEFAULT_EXTENDED_ATTR)),
): selector.BooleanSelector(selector.BooleanSelectorConfig()),
vol.Optional(
CONF_SHOW_TIME,
- default=(
- self.config_entry.data[CONF_SHOW_TIME]
- if CONF_SHOW_TIME in self.config_entry.data
- else DEFAULT_SHOW_TIME
- ),
+ default=(self.config_entry.data.get(CONF_SHOW_TIME, DEFAULT_SHOW_TIME)),
): selector.BooleanSelector(selector.BooleanSelectorConfig()),
vol.Optional(
CONF_DATE_FORMAT,
default=DEFAULT_DATE_FORMAT,
description={
- "suggested_value": self.config_entry.data[CONF_DATE_FORMAT]
- if CONF_DATE_FORMAT in self.config_entry.data
- else DEFAULT_DATE_FORMAT
+ "suggested_value": self.config_entry.data.get(
+ CONF_DATE_FORMAT, DEFAULT_DATE_FORMAT
+ )
},
): selector.SelectSelector(
selector.SelectSelectorConfig(
@@ -432,11 +419,7 @@ async def async_step_init(self, user_input=None):
),
vol.Optional(
CONF_USE_GPS,
- default=(
- self.config_entry.data[CONF_USE_GPS]
- if CONF_USE_GPS in self.config_entry.data
- else DEFAULT_USE_GPS
- ),
+ default=(self.config_entry.data.get(CONF_USE_GPS, DEFAULT_USE_GPS)),
): selector.BooleanSelector(selector.BooleanSelectorConfig()),
}
)
diff --git a/custom_components/places/const.py b/custom_components/places/const.py
index 3db7914b..ad799db5 100644
--- a/custom_components/places/const.py
+++ b/custom_components/places/const.py
@@ -1,3 +1,7 @@
+"""Constants for places."""
+
+from collections.abc import MutableMapping
+
from homeassistant.const import (
ATTR_GPS_ACCURACY,
CONF_API_KEY,
@@ -27,18 +31,18 @@
# Settings
-TRACKING_DOMAINS = [
- str(Platform.DEVICE_TRACKER),
- str("person"),
- str(Platform.SENSOR),
+TRACKING_DOMAINS: list[str] = [
+ Platform.DEVICE_TRACKER,
+ "person",
+ Platform.SENSOR,
CONF_ZONE,
"variable",
]
-TRACKING_DOMAINS_NEED_LATLONG = [
- str(Platform.SENSOR),
+TRACKING_DOMAINS_NEED_LATLONG: list[str] = [
+ Platform.SENSOR,
"variable",
]
-HOME_LOCATION_DOMAINS = [CONF_ZONE]
+HOME_LOCATION_DOMAINS: list[str] = [CONF_ZONE]
# Config
CONF_DEVICETRACKER_ID = "devicetracker_id"
@@ -115,7 +119,7 @@
# Attribute Lists
-CONFIG_ATTRIBUTES_LIST = [
+CONFIG_ATTRIBUTES_LIST: list[str] = [
CONF_API_KEY,
CONF_DEVICETRACKER_ID,
CONF_EXTENDED_ATTR,
@@ -131,7 +135,7 @@
CONF_USE_GPS,
CONF_UNIQUE_ID,
]
-RESET_ATTRIBUTE_LIST = [
+RESET_ATTRIBUTE_LIST: list[str] = [
ATTR_CITY,
ATTR_CITY_CLEAN,
ATTR_COUNTRY,
@@ -160,7 +164,7 @@
ATTR_WIKIDATA_DICT,
ATTR_WIKIDATA_ID,
]
-EXTRA_STATE_ATTRIBUTE_LIST = [
+EXTRA_STATE_ATTRIBUTE_LIST: list[str] = [
ATTR_PLACE_NAME,
ATTR_STREET_NUMBER,
ATTR_STREET,
@@ -203,7 +207,7 @@
ATTR_LAST_CHANGED,
ATTR_LAST_UPDATED,
]
-JSON_IGNORE_ATTRIBUTE_LIST = [
+JSON_IGNORE_ATTRIBUTE_LIST: list[str] = [
ATTR_ATTRIBUTES,
ATTR_DEVICETRACKER_ID,
ATTR_DISPLAY_OPTIONS,
@@ -218,7 +222,7 @@
ATTR_LOCATION_PREVIOUS,
ATTR_PREVIOUS_STATE,
]
-JSON_ATTRIBUTE_LIST = [
+JSON_ATTRIBUTE_LIST: list[str] = [
ATTR_CITY,
ATTR_CITY_CLEAN,
ATTR_COUNTRY,
@@ -264,7 +268,7 @@
ATTR_WIKIDATA_ID,
ATTR_SHOW_DATE,
]
-EVENT_ATTRIBUTE_LIST = [
+EVENT_ATTRIBUTE_LIST: list[str] = [
ATTR_PLACE_NAME,
ATTR_LAST_CHANGED,
ATTR_LAST_PLACE_NAME,
@@ -284,13 +288,13 @@
ATTR_OSM_ID,
ATTR_OSM_TYPE,
]
-EXTENDED_ATTRIBUTE_LIST = [
+EXTENDED_ATTRIBUTE_LIST: list[str] = [
ATTR_WIKIDATA_ID,
ATTR_OSM_DICT,
ATTR_OSM_DETAILS_DICT,
ATTR_WIKIDATA_DICT,
]
-PLACE_NAME_DUPLICATE_LIST = [
+PLACE_NAME_DUPLICATE_LIST: list[str] = [
ATTR_STREET,
ATTR_STREET_REF,
ATTR_PLACE_NEIGHBOURHOOD,
@@ -306,7 +310,7 @@
ATTR_DEVICETRACKER_ZONE_NAME,
]
-DISPLAY_OPTIONS_MAP = {
+DISPLAY_OPTIONS_MAP: MutableMapping[str, str] = {
"driving": ATTR_DRIVING,
"place_name": ATTR_PLACE_NAME,
"name": ATTR_PLACE_NAME,
diff --git a/custom_components/places/sensor.py b/custom_components/places/sensor.py
index c6b17591..ae73405b 100644
--- a/custom_components/places/sensor.py
+++ b/custom_components/places/sensor.py
@@ -1,5 +1,4 @@
-"""
-Place Support for OpenStreetMap Geocode sensors.
+"""Place Support for OpenStreetMap Geocode sensors.
Previous Authors: Jim Thompson, Ian Richardson
Current Author: Snuffy2
@@ -12,21 +11,25 @@
GitHub: https://github.com/custom-components/places
"""
+from collections.abc import MutableMapping
+import contextlib
import copy
+from datetime import datetime, timedelta
import json
import locale
import logging
-import os
+from pathlib import Path
import re
-from datetime import datetime, timedelta
+from typing import Any
from zoneinfo import ZoneInfo
-import homeassistant.helpers.entity_registry as er
import requests
-from homeassistant import config_entries, core
+from urllib3.exceptions import NewConnectionError
+
from homeassistant.components.recorder import DATA_INSTANCE as RECORDER_INSTANCE
from homeassistant.components.sensor import SensorEntity
from homeassistant.components.zone import ATTR_PASSIVE
+from homeassistant.config_entries import ConfigEntry
from homeassistant.const import (
ATTR_FRIENDLY_NAME,
ATTR_GPS_ACCURACY,
@@ -42,15 +45,13 @@
STATE_UNAVAILABLE,
STATE_UNKNOWN,
)
-from homeassistant.core import Event
+from homeassistant.core import Event, HomeAssistant, callback
from homeassistant.helpers.entity import generate_entity_id
-from homeassistant.helpers.event import (
- EventStateChangedData,
- async_track_state_change_event,
-)
+from homeassistant.helpers.entity_platform import AddEntitiesCallback
+import homeassistant.helpers.entity_registry as er
+from homeassistant.helpers.event import EventStateChangedData, async_track_state_change_event
from homeassistant.util import Throttle, slugify
from homeassistant.util.location import distance
-from urllib3.exceptions import NewConnectionError
from .const import (
ATTR_ATTRIBUTES,
@@ -145,127 +146,168 @@
VERSION,
)
-_LOGGER = logging.getLogger(__name__)
+_LOGGER: logging.Logger = logging.getLogger(__name__)
THROTTLE_INTERVAL = timedelta(seconds=600)
MIN_THROTTLE_INTERVAL = timedelta(seconds=10)
SCAN_INTERVAL = timedelta(seconds=30)
-PLACES_JSON_FOLDER = ""
async def async_setup_entry(
- hass: core.HomeAssistant,
- config_entry: config_entries.ConfigEntry,
- async_add_entities,
+ hass: HomeAssistant,
+ config_entry: ConfigEntry,
+ async_add_entities: AddEntitiesCallback,
) -> None:
- """Setup the sensor platform with a config_entry (config_flow)."""
- global PLACES_JSON_FOLDER
- PLACES_JSON_FOLDER = hass.config.path("custom_components", DOMAIN, "json_sensors")
- _LOGGER.debug(f"json_sensors Location: {PLACES_JSON_FOLDER}")
- await hass.async_add_executor_job(_create_json_folder)
-
+ """Create places sensor entities."""
# _LOGGER.debug(f"[aync_setup_entity] all entities: {hass.data.get(DOMAIN)}")
- config = hass.data.get(DOMAIN).get(config_entry.entry_id)
- unique_id = config_entry.entry_id
- name = config.get(CONF_NAME)
- filename = f"{DOMAIN}-{slugify(unique_id)}.json"
- imported_attributes = await hass.async_add_executor_job(
- _get_dict_from_json_file, name, filename
+ config: MutableMapping[str, Any] = dict(config_entry.data)
+ unique_id: str = config_entry.entry_id
+ name: str = config[CONF_NAME]
+ json_folder: str = hass.config.path("custom_components", DOMAIN, "json_sensors")
+ await hass.async_add_executor_job(_create_json_folder, json_folder)
+ filename: str = f"{DOMAIN}-{slugify(unique_id)}.json"
+ imported_attributes: MutableMapping[str, Any] = await hass.async_add_executor_job(
+ _get_dict_from_json_file, name, filename, json_folder
)
# _LOGGER.debug(f"[async_setup_entry] name: {name}")
# _LOGGER.debug(f"[async_setup_entry] unique_id: {unique_id}")
# _LOGGER.debug(f"[async_setup_entry] config: {config}")
if config.get(CONF_EXTENDED_ATTR, DEFAULT_EXTENDED_ATTR):
- _LOGGER.debug(
- f"({name}) Extended Attr is True. Excluding from Recorder"
- )
+ _LOGGER.debug("(%s) Extended Attr is True. Excluding from Recorder", name)
async_add_entities(
- [PlacesNoRecorder(hass, config, config_entry, name, unique_id, imported_attributes)],
+ [
+ PlacesNoRecorder(
+ hass=hass,
+ config=config,
+ config_entry=config_entry,
+ name=name,
+ unique_id=unique_id,
+ imported_attributes=imported_attributes,
+ )
+ ],
update_before_add=True,
)
else:
async_add_entities(
- [Places(hass, config, config_entry, name, unique_id, imported_attributes)],
+ [
+ Places(
+ hass=hass,
+ config=config,
+ config_entry=config_entry,
+ name=name,
+ unique_id=unique_id,
+ imported_attributes=imported_attributes,
+ )
+ ],
update_before_add=True,
)
-def _create_json_folder():
+def _create_json_folder(json_folder: str) -> None:
try:
- os.makedirs(PLACES_JSON_FOLDER, exist_ok=True)
+ Path(json_folder).mkdir(parents=True, exist_ok=True)
except OSError as e:
_LOGGER.warning(
- f"OSError creating folder for JSON sensor files: {
- e.__class__.__qualname__}: {e}"
- )
- except Exception as e:
- _LOGGER.warning(
- f"Unknown Exception creating folder for JSON sensor files: {
- e.__class__.__qualname__}: {e}"
+ "OSError creating folder for JSON sensor files: %s: %s", e.__class__.__qualname__, e
)
-def _get_dict_from_json_file(name, filename):
- sensor_attributes = {}
+def _get_dict_from_json_file(
+ name: str, filename: str, json_folder: str
+) -> MutableMapping[str, Any]:
+ sensor_attributes: MutableMapping[str, Any] = {}
try:
- with open(
- os.path.join(PLACES_JSON_FOLDER, filename),
- "r",
- ) as jsonfile:
+ json_file_path: Path = Path(json_folder) / filename
+ with json_file_path.open() as jsonfile:
sensor_attributes = json.load(jsonfile)
except OSError as e:
_LOGGER.debug(
- f"({name}) [Init] No JSON file to import "
- f"({filename}): {e.__class__.__qualname__}: {e}"
+ "(%s) [Init] No JSON file to import (%s): %s: %s",
+ name,
+ filename,
+ e.__class__.__qualname__,
+ e,
)
return {}
- except Exception as e:
+ return sensor_attributes
+
+
+def _remove_json_file(name: str, filename: str, json_folder: str) -> None:
+ try:
+ json_file_path: Path = Path(json_folder) / filename
+ json_file_path.unlink()
+ except OSError as e:
_LOGGER.debug(
- f"({name}) [Init] Unknown Exception importing JSON file "
- f"({filename}): {e.__class__.__qualname__}: {e}"
+ "(%s) OSError removing JSON sensor file (%s): %s: %s",
+ name,
+ filename,
+ e.__class__.__qualname__,
+ e,
)
- return {}
- return sensor_attributes
+ else:
+ _LOGGER.debug("(%s) JSON sensor file removed: %s", name, filename)
+
+
+def _is_float(value: Any) -> bool:
+ if value is not None:
+ try:
+ float(value)
+ except ValueError:
+ return False
+ else:
+ return True
+ return False
class Places(SensorEntity):
"""Representation of a Places Sensor."""
def __init__(
- self, hass, config, config_entry, name, unique_id, imported_attributes
- ):
+ self,
+ hass: HomeAssistant,
+ config: MutableMapping[str, Any],
+ config_entry: ConfigEntry,
+ name: str,
+ unique_id: str,
+ imported_attributes: MutableMapping[str, Any],
+ ) -> None:
"""Initialize the sensor."""
self._attr_should_poll = True
- _LOGGER.info(f"({name}) [Init] Places sensor: {name}")
- _LOGGER.debug(f"({name}) [Init] System Locale: {locale.getlocale()}")
+ _LOGGER.info("(%s) [Init] Places sensor: %s", name, name)
+ _LOGGER.debug("(%s) [Init] System Locale: %s", name, locale.getlocale())
_LOGGER.debug(
- f"({name}) [Init] System Locale Date Format: {
- str(locale.nl_langinfo(locale.D_FMT))}"
+ "(%s) [Init] System Locale Date Format: %s", name, locale.nl_langinfo(locale.D_FMT)
)
- _LOGGER.debug(f"({name}) [Init] HASS TimeZone: {hass.config.time_zone}")
+ _LOGGER.debug("(%s) [Init] HASS TimeZone: %s", name, hass.config.time_zone)
self._warn_if_device_tracker_prob = False
- self._internal_attr = {}
+ self._internal_attr: MutableMapping[str, Any] = {}
self._set_attr(ATTR_INITIAL_UPDATE, True)
- self._config = config
- self._config_entry = config_entry
- self._hass = hass
+ self._config: MutableMapping[str, Any] = config
+ self._config_entry: ConfigEntry = config_entry
+ self._hass: HomeAssistant = hass
self._set_attr(CONF_NAME, name)
- self._attr_name = name
+ self._attr_name: str = name
self._set_attr(CONF_UNIQUE_ID, unique_id)
- self._attr_unique_id = unique_id
- registry = er.async_get(self._hass)
- current_entity_id = registry.async_get_entity_id(
- PLATFORM, DOMAIN, self._attr_unique_id
- )
- if current_entity_id is not None:
- self._entity_id = current_entity_id
+ self._attr_unique_id: str = unique_id
+ registry: er.RegistryEntry | None = er.async_get(self._hass)
+ self._json_folder: str = hass.config.path("custom_components", DOMAIN, "json_sensors")
+ _LOGGER.debug("json_sensors Location: %s", self._json_folder)
+ current_entity_id: str | None = None
+ if registry:
+ current_entity_id = registry.async_get_entity_id(PLATFORM, DOMAIN, self._attr_unique_id)
+ if current_entity_id:
+ self._entity_id: str = current_entity_id
else:
self._entity_id = generate_entity_id(
ENTITY_ID_FORMAT, slugify(name.lower()), hass=self._hass
)
- _LOGGER.debug(f"({self._attr_name}) [Init] entity_id: {self._entity_id}")
+ _LOGGER.debug("(%s) [Init] entity_id: %s", self._attr_name, self._entity_id)
+ self._street_num_i: int = -1
+ self._street_i: int = -1
+ self._temp_i: int = 0
+ self._adv_options_state_list: list = []
self._set_attr(CONF_ICON, DEFAULT_ICON)
self._attr_icon = DEFAULT_ICON
self._set_attr(CONF_API_KEY, config.get(CONF_API_KEY))
@@ -273,33 +315,27 @@ def __init__(
CONF_DISPLAY_OPTIONS,
config.setdefault(CONF_DISPLAY_OPTIONS, DEFAULT_DISPLAY_OPTIONS).lower(),
)
- self._set_attr(CONF_DEVICETRACKER_ID, config.get(CONF_DEVICETRACKER_ID).lower())
+ self._set_attr(CONF_DEVICETRACKER_ID, config[CONF_DEVICETRACKER_ID].lower())
# Consider reconciling this in the future
- self._set_attr(ATTR_DEVICETRACKER_ID, config.get(CONF_DEVICETRACKER_ID).lower())
- self._set_attr(
- CONF_HOME_ZONE, config.setdefault(CONF_HOME_ZONE, DEFAULT_HOME_ZONE).lower()
- )
+ self._set_attr(ATTR_DEVICETRACKER_ID, config[CONF_DEVICETRACKER_ID].lower())
+ self._set_attr(CONF_HOME_ZONE, config.setdefault(CONF_HOME_ZONE, DEFAULT_HOME_ZONE).lower())
self._set_attr(
CONF_MAP_PROVIDER,
config.setdefault(CONF_MAP_PROVIDER, DEFAULT_MAP_PROVIDER).lower(),
)
- self._set_attr(
- CONF_MAP_ZOOM, int(config.setdefault(CONF_MAP_ZOOM, DEFAULT_MAP_ZOOM))
- )
+ self._set_attr(CONF_MAP_ZOOM, int(config.setdefault(CONF_MAP_ZOOM, DEFAULT_MAP_ZOOM)))
self._set_attr(CONF_LANGUAGE, config.get(CONF_LANGUAGE))
if not self._is_attr_blank(CONF_LANGUAGE):
self._set_attr(
CONF_LANGUAGE,
- self._get_attr(CONF_LANGUAGE).replace(" ", "").strip(),
+ self._get_attr_safe_str(CONF_LANGUAGE).replace(" ", "").strip(),
)
self._set_attr(
CONF_EXTENDED_ATTR,
config.setdefault(CONF_EXTENDED_ATTR, DEFAULT_EXTENDED_ATTR),
)
- self._set_attr(
- CONF_SHOW_TIME, config.setdefault(CONF_SHOW_TIME, DEFAULT_SHOW_TIME)
- )
+ self._set_attr(CONF_SHOW_TIME, config.setdefault(CONF_SHOW_TIME, DEFAULT_SHOW_TIME))
self._set_attr(
CONF_DATE_FORMAT,
config.setdefault(CONF_DATE_FORMAT, DEFAULT_DATE_FORMAT).lower(),
@@ -311,8 +347,9 @@ def __init__(
)
self._set_attr(ATTR_DISPLAY_OPTIONS, self._get_attr(CONF_DISPLAY_OPTIONS))
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) [Init] JSON Filename: "
- f"{self._get_attr(ATTR_JSON_FILENAME)}"
+ "(%s) [Init] JSON Filename: %s",
+ self._get_attr(CONF_NAME),
+ self._get_attr(ATTR_JSON_FILENAME),
)
self._attr_native_value = None # Represents the state in SensorEntity
@@ -320,61 +357,38 @@ def __init__(
if (
not self._is_attr_blank(CONF_HOME_ZONE)
- and CONF_LATITUDE
- in hass.states.get(self._get_attr(CONF_HOME_ZONE)).attributes
- and hass.states.get(self._get_attr(CONF_HOME_ZONE)).attributes.get(
- CONF_LATITUDE
- )
+ and CONF_LATITUDE in hass.states.get(self._get_attr(CONF_HOME_ZONE)).attributes
+ and hass.states.get(self._get_attr(CONF_HOME_ZONE)).attributes.get(CONF_LATITUDE)
is not None
- and self._is_float(
- hass.states.get(self._get_attr(CONF_HOME_ZONE)).attributes.get(
- CONF_LATITUDE
- )
+ and _is_float(
+ hass.states.get(self._get_attr(CONF_HOME_ZONE)).attributes.get(CONF_LATITUDE)
)
):
self._set_attr(
ATTR_HOME_LATITUDE,
- str(
- hass.states.get(self._get_attr(CONF_HOME_ZONE)).attributes.get(
- CONF_LATITUDE
- )
- ),
+ str(hass.states.get(self._get_attr(CONF_HOME_ZONE)).attributes.get(CONF_LATITUDE)),
)
if (
not self._is_attr_blank(CONF_HOME_ZONE)
- and CONF_LONGITUDE
- in hass.states.get(self._get_attr(CONF_HOME_ZONE)).attributes
- and hass.states.get(self._get_attr(CONF_HOME_ZONE)).attributes.get(
- CONF_LONGITUDE
- )
+ and CONF_LONGITUDE in hass.states.get(self._get_attr(CONF_HOME_ZONE)).attributes
+ and hass.states.get(self._get_attr(CONF_HOME_ZONE)).attributes.get(CONF_LONGITUDE)
is not None
- and self._is_float(
- hass.states.get(self._get_attr(CONF_HOME_ZONE)).attributes.get(
- CONF_LONGITUDE
- )
+ and _is_float(
+ hass.states.get(self._get_attr(CONF_HOME_ZONE)).attributes.get(CONF_LONGITUDE)
)
):
self._set_attr(
ATTR_HOME_LONGITUDE,
- str(
- hass.states.get(self._get_attr(CONF_HOME_ZONE)).attributes.get(
- CONF_LONGITUDE
- )
- ),
+ str(hass.states.get(self._get_attr(CONF_HOME_ZONE)).attributes.get(CONF_LONGITUDE)),
)
self._attr_entity_picture = (
- hass.states.get(self._get_attr(CONF_DEVICETRACKER_ID)).attributes.get(
- ATTR_PICTURE
- )
+ hass.states.get(self._get_attr(CONF_DEVICETRACKER_ID)).attributes.get(ATTR_PICTURE)
if hass.states.get(self._get_attr(CONF_DEVICETRACKER_ID))
else None
)
self._set_attr(ATTR_SHOW_DATE, False)
- # self._set_attr(ATTR_UPDATES_SKIPPED, 0)
- # sensor_attributes = self._hass.async_add_executor_job(self._get_dict_from_json_file, self._get_attr(CONF_NAME), self._get_attr(ATTR_JSON_FILENAME),)
- # _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) [Init] Sensor Attributes to Import: {imported_attributes}")
self._import_attributes_from_json(imported_attributes)
##
# For debugging:
@@ -386,28 +400,29 @@ def __init__(
##
if not self._get_attr(ATTR_INITIAL_UPDATE):
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) "
- "[Init] Sensor Attributes Imported from JSON file"
+ "(%s) [Init] Sensor Attributes Imported from JSON file", self._get_attr(CONF_NAME)
)
self._cleanup_attributes()
if self._get_attr(CONF_EXTENDED_ATTR):
self._exclude_event_types()
_LOGGER.info(
- f"({self._get_attr(CONF_NAME)}) [Init] Tracked Entity ID: "
- f"{self._get_attr(CONF_DEVICETRACKER_ID)}"
+ "(%s) [Init] Tracked Entity ID: %s",
+ self._get_attr(CONF_NAME),
+ self._get_attr(CONF_DEVICETRACKER_ID),
)
- def _exclude_event_types(self):
+ def _exclude_event_types(self) -> None:
if RECORDER_INSTANCE in self._hass.data:
ha_history_recorder = self._hass.data[RECORDER_INSTANCE]
ha_history_recorder.exclude_event_types.add(EVENT_TYPE)
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) "
- f"exclude_event_types: {ha_history_recorder.exclude_event_types}"
+ "(%s) exclude_event_types: %s",
+ self._get_attr(CONF_NAME),
+ ha_history_recorder.exclude_event_types,
)
async def async_added_to_hass(self) -> None:
- """Added to hass."""
+ """Run after sensor is added to HA."""
await super().async_added_to_hass()
self.async_on_remove(
async_track_state_change_event(
@@ -417,39 +432,23 @@ async def async_added_to_hass(self) -> None:
)
)
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) "
- "[Init] Subscribed to Tracked Entity state change events"
+ "(%s) [Init] Subscribed to Tracked Entity state change events",
+ self._get_attr(CONF_NAME),
)
- def _remove_json_file(self, name, filename):
- try:
- os.remove(os.path.join(PLACES_JSON_FOLDER, filename))
- except OSError as e:
- _LOGGER.debug(
- f"({name}) OSError removing JSON sensor file "
- f"({filename}): {e.__class__.__qualname__}: {e}"
- )
- except Exception as e:
- _LOGGER.debug(
- f"({name}) Unknown Exception removing JSON sensor file "
- f"({filename}): {e.__class__.__qualname__}: {e}"
- )
- else:
- _LOGGER.debug(f"({name}) JSON sensor file removed: {filename}")
-
async def async_will_remove_from_hass(self) -> None:
"""Run when entity will be removed from hass."""
await self._hass.async_add_executor_job(
- self._remove_json_file,
+ _remove_json_file,
self._get_attr(CONF_NAME),
self._get_attr(ATTR_JSON_FILENAME),
+ self._json_folder,
)
if RECORDER_INSTANCE in self._hass.data and self._get_attr(CONF_EXTENDED_ATTR):
_LOGGER.debug(
- f"({self._attr_name}) Removing entity exclusion from recorder: "
- f"{self._entity_id}"
+ "(%s) Removing entity exclusion from recorder: %s", self._attr_name, self._entity_id
)
# Only do this if no places entities with extended_attr exist
ex_attr_count = 0
@@ -457,20 +456,19 @@ async def async_will_remove_from_hass(self) -> None:
if ent.get(CONF_EXTENDED_ATTR):
ex_attr_count += 1
- if (
- self._get_attr(CONF_EXTENDED_ATTR) and ex_attr_count == 1
- ) or ex_attr_count == 0:
+ if (self._get_attr(CONF_EXTENDED_ATTR) and ex_attr_count == 1) or ex_attr_count == 0:
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) "
- f"Removing event exclusion from recorder: {EVENT_TYPE}"
+ "(%s) Removing event exclusion from recorder: %s",
+ self._get_attr(CONF_NAME),
+ EVENT_TYPE,
)
ha_history_recorder = self._hass.data[RECORDER_INSTANCE]
ha_history_recorder.exclude_event_types.discard(EVENT_TYPE)
@property
- def extra_state_attributes(self):
+ def extra_state_attributes(self) -> dict[str, Any]:
"""Return the state attributes."""
- return_attr = {}
+ return_attr: dict[str, Any] = {}
self._cleanup_attributes()
for attr in EXTRA_STATE_ATTRIBUTE_LIST:
if self._get_attr(attr):
@@ -483,9 +481,9 @@ def extra_state_attributes(self):
# _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) Extra State Attributes: {return_attr}")
return return_attr
- def _import_attributes_from_json(self, json_attr=None):
+ def _import_attributes_from_json(self, json_attr: MutableMapping[str, Any]) -> None:
"""Import the JSON state attributes. Takes a Dictionary as input."""
- if json_attr is None or not isinstance(json_attr, dict) or not json_attr:
+ if not isinstance(json_attr, MutableMapping):
return
self._set_attr(ATTR_INITIAL_UPDATE, False)
@@ -501,45 +499,61 @@ def _import_attributes_from_json(self, json_attr=None):
json_attr.pop(attr, None)
if json_attr is not None and json_attr:
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) "
- f"[import_attributes] Attributes not imported: {json_attr}"
+ "(%s) [import_attributes] Attributes not imported: %s",
+ self._get_attr(CONF_NAME),
+ json_attr,
)
- def _cleanup_attributes(self):
+ def _cleanup_attributes(self) -> None:
for attr in list(self._internal_attr):
if self._is_attr_blank(attr):
self._clear_attr(attr)
- def _is_attr_blank(self, attr):
+ def _is_attr_blank(self, attr: str) -> bool:
if self._internal_attr.get(attr) or self._internal_attr.get(attr) == 0:
return False
- else:
- return True
+ return True
- def _get_attr(self, attr, default=None):
+ def _get_attr(self, attr: str | None, default: Any | None = None) -> None | Any:
if attr is None or (default is None and self._is_attr_blank(attr)):
return None
- else:
- return self._internal_attr.get(attr, default)
-
- def _set_attr(self, attr, value=None):
- if attr is not None:
- self._internal_attr.update({attr: value})
+ return self._internal_attr.get(attr, default)
- def _clear_attr(self, attr):
- self._internal_attr.pop(attr, None)
-
- def _is_float(self, value):
+ def _get_attr_safe_str(self, attr: str | None, default: Any | None = None) -> str:
+ value = self._get_attr(attr=attr, default=default)
if value is not None:
try:
- float(value)
- return True
+ return str(value)
except ValueError:
- return False
- else:
- return False
+ return ""
+ return ""
+
+ def _get_attr_safe_float(self, attr: str | None, default: Any | None = None) -> float:
+ value = self._get_attr(attr=attr, default=default)
+ if not isinstance(value, float):
+ return 0
+ return value
+
+ def _get_attr_safe_list(self, attr: str | None, default: Any | None = None) -> list:
+ value = self._get_attr(attr=attr, default=default)
+ if not isinstance(value, list):
+ return []
+ return value
+
+ def _get_attr_safe_dict(self, attr: str | None, default: Any | None = None) -> MutableMapping:
+ value = self._get_attr(attr=attr, default=default)
+ if not isinstance(value, MutableMapping):
+ return {}
+ return value
+
+ def _set_attr(self, attr: str, value: Any | None = None) -> None:
+ if attr:
+ self._internal_attr.update({attr: value})
- async def _async_is_devicetracker_set(self):
+ def _clear_attr(self, attr: str) -> None:
+ self._internal_attr.pop(attr, None)
+
+ async def _async_is_devicetracker_set(self) -> int:
proceed_with_update = 0
# 0: False. 1: True. 2: False, but set direction of travel to stationary
@@ -552,21 +566,23 @@ async def _async_is_devicetracker_set(self):
str,
)
and self._hass.states.get(self._get_attr(CONF_DEVICETRACKER_ID)).lower()
- in ["none", STATE_UNKNOWN, STATE_UNAVAILABLE]
+ in {"none", STATE_UNKNOWN, STATE_UNAVAILABLE}
)
):
if self._warn_if_device_tracker_prob or self._get_attr(ATTR_INITIAL_UPDATE):
_LOGGER.warning(
- f"({self._get_attr(CONF_NAME)}) Tracked Entity "
- f"({self._get_attr(CONF_DEVICETRACKER_ID)}) "
- "is not set or is not available. Not Proceeding with Update."
+ "(%s) Tracked Entity (%s) "
+ "is not set or is not available. Not Proceeding with Update",
+ self._get_attr(CONF_NAME),
+ self._get_attr(CONF_DEVICETRACKER_ID),
)
self._warn_if_device_tracker_prob = False
else:
_LOGGER.info(
- f"({self._get_attr(CONF_NAME)}) Tracked Entity "
- f"({self._get_attr(CONF_DEVICETRACKER_ID)}) "
- "is not set or is not available. Not Proceeding with Update."
+ "(%s) Tracked Entity (%s) "
+ "is not set or is not available. Not Proceeding with Update",
+ self._get_attr(CONF_NAME),
+ self._get_attr(CONF_DEVICETRACKER_ID),
)
return 0
# 0: False. 1: True. 2: False, but set direction of travel to stationary
@@ -579,23 +595,23 @@ async def _async_is_devicetracker_set(self):
in self._hass.states.get(self._get_attr(CONF_DEVICETRACKER_ID)).attributes
and CONF_LONGITUDE
in self._hass.states.get(self._get_attr(CONF_DEVICETRACKER_ID)).attributes
- and self._hass.states.get(
- self._get_attr(CONF_DEVICETRACKER_ID)
- ).attributes.get(CONF_LATITUDE)
+ and self._hass.states.get(self._get_attr(CONF_DEVICETRACKER_ID)).attributes.get(
+ CONF_LATITUDE
+ )
is not None
- and self._hass.states.get(
- self._get_attr(CONF_DEVICETRACKER_ID)
- ).attributes.get(CONF_LONGITUDE)
+ and self._hass.states.get(self._get_attr(CONF_DEVICETRACKER_ID)).attributes.get(
+ CONF_LONGITUDE
+ )
is not None
- and self._is_float(
- self._hass.states.get(
- self._get_attr(CONF_DEVICETRACKER_ID)
- ).attributes.get(CONF_LATITUDE)
+ and _is_float(
+ self._hass.states.get(self._get_attr(CONF_DEVICETRACKER_ID)).attributes.get(
+ CONF_LATITUDE
+ )
)
- and self._is_float(
- self._hass.states.get(
- self._get_attr(CONF_DEVICETRACKER_ID)
- ).attributes.get(CONF_LONGITUDE)
+ and _is_float(
+ self._hass.states.get(self._get_attr(CONF_DEVICETRACKER_ID)).attributes.get(
+ CONF_LONGITUDE
+ )
)
):
self._warn_if_device_tracker_prob = True
@@ -604,119 +620,115 @@ async def _async_is_devicetracker_set(self):
else:
if self._warn_if_device_tracker_prob or self._get_attr(ATTR_INITIAL_UPDATE):
_LOGGER.warning(
- f"({self._get_attr(CONF_NAME)}) Tracked Entity "
- f"({self._get_attr(CONF_DEVICETRACKER_ID)}) "
- "Latitude/Longitude is not set or is not a number. "
- "Not Proceeding with Update."
+ "(%s) Tracked Entity (%s) "
+ "Latitude/Longitude is not set or is not a number. Not Proceeding with Update.",
+ self._get_attr(CONF_NAME),
+ self._get_attr(CONF_DEVICETRACKER_ID),
)
self._warn_if_device_tracker_prob = False
else:
_LOGGER.info(
- f"({self._get_attr(CONF_NAME)}) Tracked Entity "
- f"({self._get_attr(CONF_DEVICETRACKER_ID)}) "
- "Latitude/Longitude is not set or is not a number. "
- "Not Proceeding with Update."
+ "(%s) Tracked Entity (%s) "
+ "Latitude/Longitude is not set or is not a number. Not Proceeding with Update.",
+ self._get_attr(CONF_NAME),
+ self._get_attr(CONF_DEVICETRACKER_ID),
)
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) Tracked Entity "
- f"({self._get_attr(CONF_DEVICETRACKER_ID)}) details: "
- f"{self._hass.states.get(self._get_attr(CONF_DEVICETRACKER_ID))}"
+ "(%s) Tracked Entity (%s) details: %s",
+ self._get_attr(CONF_NAME),
+ self._get_attr(CONF_DEVICETRACKER_ID),
+ self._hass.states.get(self._get_attr(CONF_DEVICETRACKER_ID)),
)
return 0
# 0: False. 1: True. 2: False, but set direction of travel to stationary
return proceed_with_update
@Throttle(MIN_THROTTLE_INTERVAL)
- @core.callback
- def _async_tsc_update(self, event: Event[EventStateChangedData]):
- """Call the _async_do_update function based on the TSC (track state change) event"""
+ @callback
+ def _async_tsc_update(self, event: Event[EventStateChangedData]) -> None:
+ """Call the _async_do_update function based on the TSC (track state change) event."""
# _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) [TSC Update] event: {event}")
new_state = event.data["new_state"]
if new_state is None or (
isinstance(new_state.state, str)
- and new_state.state.lower() in ["none", STATE_UNKNOWN, STATE_UNAVAILABLE]
+ and new_state.state.lower() in {"none", STATE_UNKNOWN, STATE_UNAVAILABLE}
):
return
# _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) [TSC Update] new_state: {new_state}")
- update_type = "Track State Change"
+ update_type: str = "Track State Change"
self._hass.async_create_task(self._async_do_update(update_type))
@Throttle(THROTTLE_INTERVAL)
- async def async_update(self):
- """Call the _async_do_update function based on scan interval and throttle"""
+ async def async_update(self) -> None:
+ """Call the _async_do_update function based on scan interval and throttle."""
update_type = "Scan Interval"
self._hass.async_create_task(self._async_do_update(update_type))
- async def _async_clear_since_from_state(self, orig_state):
+ @staticmethod
+ async def _async_clear_since_from_state(orig_state: str) -> str:
return re.sub(r" \(since \d\d[:/]\d\d\)", "", orig_state)
- async def _async_in_zone(self):
+ async def _async_in_zone(self) -> bool:
if not self._is_attr_blank(ATTR_DEVICETRACKER_ZONE):
zone_state = self._hass.states.get(
- f"{CONF_ZONE}.{(self._get_attr(ATTR_DEVICETRACKER_ZONE)).lower()}"
+ f"{CONF_ZONE}.{self._get_attr_safe_str(ATTR_DEVICETRACKER_ZONE).lower()}"
)
- if (self._get_attr(CONF_DEVICETRACKER_ID)).split(".")[0] == CONF_ZONE:
- return False
- elif (
- "stationary" in (self._get_attr(ATTR_DEVICETRACKER_ZONE)).lower()
- or (self._get_attr(ATTR_DEVICETRACKER_ZONE))
- .lower()
- .startswith("statzon")
- or (self._get_attr(ATTR_DEVICETRACKER_ZONE))
- .lower()
- .startswith("ic3_statzone_")
- or (self._get_attr(ATTR_DEVICETRACKER_ZONE)).lower() == "away"
- or (self._get_attr(ATTR_DEVICETRACKER_ZONE)).lower() == "not_home"
- or (self._get_attr(ATTR_DEVICETRACKER_ZONE)).lower() == "notset"
- or (self._get_attr(ATTR_DEVICETRACKER_ZONE)).lower() == "not_set"
- ):
- return False
- elif (
- zone_state is not None
- and zone_state.attributes.get(ATTR_PASSIVE, False) is True
+ if (
+ self._get_attr_safe_str(CONF_DEVICETRACKER_ID).split(".")[0] == CONF_ZONE
+ or (
+ "stationary" in self._get_attr_safe_str(ATTR_DEVICETRACKER_ZONE).lower()
+ or self._get_attr_safe_str(ATTR_DEVICETRACKER_ZONE)
+ .lower()
+ .startswith("statzon")
+ or self._get_attr_safe_str(ATTR_DEVICETRACKER_ZONE)
+ .lower()
+ .startswith("ic3_statzone_")
+ or self._get_attr_safe_str(ATTR_DEVICETRACKER_ZONE).lower() == "away"
+ or self._get_attr_safe_str(ATTR_DEVICETRACKER_ZONE).lower() == "not_home"
+ or self._get_attr_safe_str(ATTR_DEVICETRACKER_ZONE).lower() == "notset"
+ or self._get_attr_safe_str(ATTR_DEVICETRACKER_ZONE).lower() == "not_set"
+ )
+ or (
+ zone_state is not None
+ and zone_state.attributes.get(ATTR_PASSIVE, False) is True
+ )
):
return False
- else:
- return True
- else:
- return False
+ return True
+ return False
- async def _async_cleanup_attributes(self):
- for attr in list(self._internal_attr):
+ async def _async_cleanup_attributes(self) -> None:
+ attrs: MutableMapping[str, Any] = copy.deepcopy(self._internal_attr)
+ for attr in attrs:
if self._is_attr_blank(attr):
self._clear_attr(attr)
- async def _async_check_for_updated_entity_name(self):
+ async def _async_check_for_updated_entity_name(self) -> None:
if hasattr(self, "entity_id") and self._entity_id is not None:
# _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) Entity ID: {self._entity_id}")
if (
self._hass.states.get(str(self._entity_id)) is not None
- and self._hass.states.get(str(self._entity_id)).attributes.get(
- ATTR_FRIENDLY_NAME
- )
+ and self._hass.states.get(str(self._entity_id)).attributes.get(ATTR_FRIENDLY_NAME)
is not None
and self._get_attr(CONF_NAME)
- != self._hass.states.get(str(self._entity_id)).attributes.get(
- ATTR_FRIENDLY_NAME
- )
+ != self._hass.states.get(str(self._entity_id)).attributes.get(ATTR_FRIENDLY_NAME)
):
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) Sensor Name Changed. Updating Name to: "
- f"{self._hass.states.get(
- str(self._entity_id)).attributes.get(ATTR_FRIENDLY_NAME)}"
+ "(%s) Sensor Name Changed. Updating Name to: %s",
+ self._get_attr(CONF_NAME),
+ self._hass.states.get(str(self._entity_id)).attributes.get(ATTR_FRIENDLY_NAME),
)
self._set_attr(
CONF_NAME,
- self._hass.states.get(str(self._entity_id)).attributes.get(
- ATTR_FRIENDLY_NAME
- ),
+ self._hass.states.get(str(self._entity_id)).attributes.get(ATTR_FRIENDLY_NAME),
)
self._config.update({CONF_NAME: self._get_attr(CONF_NAME)})
self._set_attr(CONF_NAME, self._get_attr(CONF_NAME))
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) Updated Config Name: "
- f"{self._config.get(CONF_NAME, None)}"
+ "(%s) Updated Config Name: %s",
+ self._get_attr(CONF_NAME),
+ self._config.get(CONF_NAME),
)
self._hass.config_entries.async_update_entry(
self._config_entry,
@@ -724,43 +736,35 @@ async def _async_check_for_updated_entity_name(self):
options=self._config_entry.options,
)
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) Updated ConfigEntry Name: "
- f"{self._config_entry.data.get(CONF_NAME)}"
+ "(%s) Updated ConfigEntry Name: %s",
+ self._get_attr(CONF_NAME),
+ self._config_entry.data.get(CONF_NAME),
)
- async def _async_get_zone_details(self):
- if (self._get_attr(CONF_DEVICETRACKER_ID)).split(".")[0] != CONF_ZONE:
+ async def _async_get_zone_details(self) -> None:
+ if self._get_attr_safe_str(CONF_DEVICETRACKER_ID).split(".")[0] != CONF_ZONE:
self._set_attr(
ATTR_DEVICETRACKER_ZONE,
self._hass.states.get(self._get_attr(CONF_DEVICETRACKER_ID)).state,
)
if await self._async_in_zone():
devicetracker_zone_name_state = None
- devicetracker_zone_id = self._hass.states.get(
+ devicetracker_zone_id: str | None = self._hass.states.get(
self._get_attr(CONF_DEVICETRACKER_ID)
).attributes.get(CONF_ZONE)
- if devicetracker_zone_id is not None:
+ if devicetracker_zone_id:
devicetracker_zone_id = f"{CONF_ZONE}.{devicetracker_zone_id}"
- devicetracker_zone_name_state = self._hass.states.get(
- devicetracker_zone_id
- )
+ devicetracker_zone_name_state = self._hass.states.get(devicetracker_zone_id)
# _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) Tracked Entity Zone ID: {devicetracker_zone_id}")
# _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) Tracked Entity Zone Name State: {devicetracker_zone_name_state}")
- if devicetracker_zone_name_state is not None:
- if (
- devicetracker_zone_name_state.attributes.get(CONF_FRIENDLY_NAME)
- is not None
- ):
+ if devicetracker_zone_name_state:
+ if devicetracker_zone_name_state.attributes.get(CONF_FRIENDLY_NAME):
self._set_attr(
ATTR_DEVICETRACKER_ZONE_NAME,
- devicetracker_zone_name_state.attributes.get(
- CONF_FRIENDLY_NAME
- ),
+ devicetracker_zone_name_state.attributes.get(CONF_FRIENDLY_NAME),
)
else:
- self._set_attr(
- ATTR_DEVICETRACKER_ZONE_NAME, devicetracker_zone_name_state.name
- )
+ self._set_attr(ATTR_DEVICETRACKER_ZONE_NAME, devicetracker_zone_name_state.name)
else:
self._set_attr(
ATTR_DEVICETRACKER_ZONE_NAME,
@@ -768,157 +772,174 @@ async def _async_get_zone_details(self):
)
if not self._is_attr_blank(ATTR_DEVICETRACKER_ZONE_NAME) and (
- self._get_attr(ATTR_DEVICETRACKER_ZONE_NAME)
+ self._get_attr_safe_str(ATTR_DEVICETRACKER_ZONE_NAME)
).lower() == self._get_attr(ATTR_DEVICETRACKER_ZONE_NAME):
self._set_attr(
ATTR_DEVICETRACKER_ZONE_NAME,
- (self._get_attr(ATTR_DEVICETRACKER_ZONE_NAME)).title(),
+ self._get_attr_safe_str(ATTR_DEVICETRACKER_ZONE_NAME).title(),
)
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) Tracked Entity Zone Name: "
- f"{self._get_attr(ATTR_DEVICETRACKER_ZONE_NAME)}"
+ "(%s) Tracked Entity Zone Name: %s",
+ self._get_attr(CONF_NAME),
+ self._get_attr(ATTR_DEVICETRACKER_ZONE_NAME),
)
else:
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) Tracked Entity Zone: "
- f"{self._get_attr(ATTR_DEVICETRACKER_ZONE)}"
+ "(%s) Tracked Entity Zone: %s",
+ self._get_attr(CONF_NAME),
+ self._get_attr(ATTR_DEVICETRACKER_ZONE),
)
self._set_attr(
ATTR_DEVICETRACKER_ZONE_NAME,
self._get_attr(ATTR_DEVICETRACKER_ZONE),
)
- async def _async_determine_if_update_needed(self):
+ async def _async_determine_if_update_needed(self) -> int:
proceed_with_update = 1
# 0: False. 1: True. 2: False, but set direction of travel to stationary
if self._get_attr(ATTR_INITIAL_UPDATE):
- _LOGGER.info(
- f"({self._get_attr(CONF_NAME)}) Performing Initial Update for user..."
- )
+ _LOGGER.info("(%s) Performing Initial Update for user", self._get_attr(CONF_NAME))
# 0: False. 1: True. 2: False, but set direction of travel to stationary
return 1
- elif self._is_attr_blank(ATTR_NATIVE_VALUE) or (
+ if self._is_attr_blank(ATTR_NATIVE_VALUE) or (
isinstance(self._get_attr(ATTR_NATIVE_VALUE), str)
- and (self._get_attr(ATTR_NATIVE_VALUE)).lower()
- in ["none", STATE_UNKNOWN, STATE_UNAVAILABLE]
+ and self._get_attr_safe_str(ATTR_NATIVE_VALUE).lower()
+ in {"none", STATE_UNKNOWN, STATE_UNAVAILABLE}
):
_LOGGER.info(
- f"({self._get_attr(CONF_NAME)}) Previous State is Unknown, performing update."
+ "(%s) Previous State is Unknown, performing update", self._get_attr(CONF_NAME)
)
# 0: False. 1: True. 2: False, but set direction of travel to stationary
return 1
- elif self._get_attr(ATTR_LOCATION_CURRENT) == self._get_attr(
- ATTR_LOCATION_PREVIOUS
- ):
+ if self._get_attr(ATTR_LOCATION_CURRENT) == self._get_attr(ATTR_LOCATION_PREVIOUS):
_LOGGER.info(
- f"({self._get_attr(CONF_NAME)}) "
- "Not performing update because coordinates are identical"
+ "(%s) Not performing update because coordinates are identical",
+ self._get_attr(CONF_NAME),
)
return 2
# 0: False. 1: True. 2: False, but set direction of travel to stationary
- elif int(self._get_attr(ATTR_DISTANCE_TRAVELED_M)) < 10:
+ if int(self._get_attr_safe_float(ATTR_DISTANCE_TRAVELED_M)) < 10:
_LOGGER.info(
- f"({self._get_attr(CONF_NAME)}) "
- "Not performing update, distance traveled from last update is less than 10 m ("
- f"{round(self._get_attr(ATTR_DISTANCE_TRAVELED_M), 1)} m)"
+ "(%s) "
+ "Not performing update, distance traveled from last update is less than 10 m (%s m)",
+ self._get_attr(CONF_NAME),
+ round(self._get_attr_safe_float(ATTR_DISTANCE_TRAVELED_M), 1),
)
return 2
# 0: False. 1: True. 2: False, but set direction of travel to stationary
return proceed_with_update
# 0: False. 1: True. 2: False, but set direction of travel to stationary
- def _get_dict_from_url(self, url, name, dict_name):
- get_dict = {}
- _LOGGER.info(f"({self._get_attr(CONF_NAME)}) Requesting data for {name}")
- _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) {name} URL: {url}")
+ def _get_dict_from_url(self, url: str, name: str, dict_name: str) -> None:
+ get_dict: MutableMapping[str, Any] = {}
+ _LOGGER.info("(%s) Requesting data for %s", self._get_attr(CONF_NAME), name)
+ _LOGGER.debug("(%s) {name} URL: %s", self._get_attr(CONF_NAME), url)
self._set_attr(dict_name, {})
- headers = {"user-agent": f"Mozilla/5.0 (Home Assistant) {DOMAIN}/{VERSION}"}
+ headers: dict[str, str] = {"user-agent": f"Mozilla/5.0 (Home Assistant) {DOMAIN}/{VERSION}"}
try:
- get_response = requests.get(url, headers=headers)
+ get_response: requests.Response | None = requests.get(url, headers=headers)
except requests.exceptions.RetryError as e:
get_response = None
_LOGGER.warning(
- f"({self._get_attr(CONF_NAME)}) Retry Error connecting to "
- f"{name} [{e.__class__.__qualname__}: {e}]: {url}"
+ "(%s) Retry Error connecting to %s [%s: %s]: %s",
+ self._get_attr(CONF_NAME),
+ name,
+ e.__class__.__qualname__,
+ e,
+ url,
)
return
except requests.exceptions.ConnectionError as e:
get_response = None
_LOGGER.warning(
- f"({self._get_attr(CONF_NAME)}) Connection Error connecting to "
- f"{name} [{e.__class__.__qualname__}: {e}]: {url}"
+ "(%s) Connection Error connecting to %s [%s: %s]: %s",
+ self._get_attr(CONF_NAME),
+ name,
+ e.__class__.__qualname__,
+ e,
+ url,
)
return
except requests.exceptions.HTTPError as e:
get_response = None
_LOGGER.warning(
- f"({self._get_attr(CONF_NAME)}) HTTP Error connecting to "
- f"{name} [{e.__class__.__qualname__}: {e}]: {url}"
+ "(%s) HTTP Error connecting to %s [%s: %s]: %s",
+ self._get_attr(CONF_NAME),
+ name,
+ e.__class__.__qualname__,
+ e,
+ url,
)
return
except requests.exceptions.Timeout as e:
get_response = None
_LOGGER.warning(
- f"({self._get_attr(CONF_NAME)}) Timeout connecting to "
- f"{name} [{e.__class__.__qualname__}: {e}]: {url}"
+ "(%s) Timeout connecting to %s [%s: %s]: %s",
+ self._get_attr(CONF_NAME),
+ name,
+ e.__class__.__qualname__,
+ e,
+ url,
)
return
except OSError as e:
# Includes error code 101, network unreachable
get_response = None
_LOGGER.warning(
- f"({self._get_attr(CONF_NAME)}) "
- f"Network unreachable error when connecting to {name} "
- f"[{e.__class__.__qualname__}: {e}]: {url}"
+ "(%s) Network unreachable error when connecting to %s [%s: %s]: %s",
+ self._get_attr(CONF_NAME),
+ name,
+ e.__class__.__qualname__,
+ e,
+ url,
)
return
except NewConnectionError as e:
get_response = None
_LOGGER.warning(
- f"({self._get_attr(CONF_NAME)}) "
- f"New Connection Error connecting to {name} "
- f"[{e.__class__.__qualname__}: {e}]: {url}"
- )
- return
- except Exception as e:
- get_response = None
- _LOGGER.warning(
- f"({self._get_attr(CONF_NAME)}) "
- f"Unknown Exception connecting to {name} "
- f"[{e.__class__.__qualname__}: {e}]: {url}"
+ "(%s) New Connection Error connecting to %s [%s: %s]: %s",
+ self._get_attr(CONF_NAME),
+ name,
+ e.__class__.__qualname__,
+ e,
+ url,
)
return
- get_json_input = {}
- if get_response is not None and get_response:
+ get_json_input: str | None = None
+ if get_response:
get_json_input = get_response.text
- _LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) {name} Response: {get_json_input}"
- )
+ _LOGGER.debug("(%s) %s Response: %s", self._get_attr(CONF_NAME), name, get_json_input)
- if get_json_input is not None and get_json_input:
+ if get_json_input:
try:
get_dict = json.loads(get_json_input)
except json.decoder.JSONDecodeError as e:
_LOGGER.warning(
- f"({self._get_attr(CONF_NAME)}) JSON Decode Error with {name} info "
- f"[{e.__class__.__qualname__}: {e}]: {get_json_input}"
+ "(%s) JSON Decode Error with %s info [%s: %s]: %s",
+ self._get_attr(CONF_NAME),
+ name,
+ e.__class__.__qualname__,
+ e,
+ get_json_input,
)
return
if "error_message" in get_dict:
_LOGGER.warning(
- f"({self._get_attr(CONF_NAME)}) An error occurred contacting the web service for "
- f"{name}: {get_dict.get('error_message')}"
+ "(%s) An error occurred contacting the web service for %s: %s",
+ self._get_attr(CONF_NAME),
+ name,
+ get_dict.get("error_message"),
)
return
if (
isinstance(get_dict, list)
and len(get_dict) == 1
- and isinstance(get_dict[0], dict)
+ and isinstance(get_dict[0], MutableMapping)
):
self._set_attr(dict_name, get_dict[0])
return
@@ -926,7 +947,7 @@ def _get_dict_from_url(self, url, name, dict_name):
self._set_attr(dict_name, get_dict)
return
- async def _async_get_map_link(self):
+ async def _async_get_map_link(self) -> None:
if self._get_attr(CONF_MAP_PROVIDER) == "google":
self._set_attr(
ATTR_MAP_LINK,
@@ -945,8 +966,8 @@ async def _async_get_map_link(self):
f"{self._get_attr(ATTR_LATITUDE)}"
f"&mlon={self._get_attr(ATTR_LONGITUDE)}"
f"#map={self._get_attr(CONF_MAP_ZOOM)}/"
- f"{str(self._get_attr(ATTR_LATITUDE))[:8]}/"
- f"{str(self._get_attr(ATTR_LONGITUDE))[:9]}"
+ f"{self._get_attr_safe_str(ATTR_LATITUDE)[:8]}/"
+ f"{self._get_attr_safe_str(ATTR_LONGITUDE)[:9]}"
),
)
else:
@@ -959,42 +980,41 @@ async def _async_get_map_link(self):
),
)
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) Map Link Type: "
- f"{self._get_attr(CONF_MAP_PROVIDER)}"
+ "(%s) Map Link Type: %s", self._get_attr(CONF_NAME), self._get_attr(CONF_MAP_PROVIDER)
)
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) Map Link URL: "
- f"{self._get_attr(ATTR_MAP_LINK)}"
+ "(%s) Map Link URL: %s", self._get_attr(CONF_NAME), self._get_attr(ATTR_MAP_LINK)
)
- async def _async_get_gps_accuracy(self):
+ async def _async_get_gps_accuracy(self) -> int:
if (
self._hass.states.get(self._get_attr(CONF_DEVICETRACKER_ID))
and self._hass.states.get(self._get_attr(CONF_DEVICETRACKER_ID)).attributes
and ATTR_GPS_ACCURACY
in self._hass.states.get(self._get_attr(CONF_DEVICETRACKER_ID)).attributes
- and self._hass.states.get(
- self._get_attr(CONF_DEVICETRACKER_ID)
- ).attributes.get(ATTR_GPS_ACCURACY)
+ and self._hass.states.get(self._get_attr(CONF_DEVICETRACKER_ID)).attributes.get(
+ ATTR_GPS_ACCURACY
+ )
is not None
- and self._is_float(
- self._hass.states.get(
- self._get_attr(CONF_DEVICETRACKER_ID)
- ).attributes.get(ATTR_GPS_ACCURACY)
+ and _is_float(
+ self._hass.states.get(self._get_attr(CONF_DEVICETRACKER_ID)).attributes.get(
+ ATTR_GPS_ACCURACY
+ )
)
):
self._set_attr(
ATTR_GPS_ACCURACY,
float(
- self._hass.states.get(
- self._get_attr(CONF_DEVICETRACKER_ID)
- ).attributes.get(ATTR_GPS_ACCURACY)
+ self._hass.states.get(self._get_attr(CONF_DEVICETRACKER_ID)).attributes.get(
+ ATTR_GPS_ACCURACY
+ )
),
)
else:
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) GPS Accuracy attribute not found in: "
- f"{self._get_attr(CONF_DEVICETRACKER_ID)}"
+ "(%s) GPS Accuracy attribute not found in: %s",
+ self._get_attr(CONF_NAME),
+ self._get_attr(CONF_DEVICETRACKER_ID),
)
proceed_with_update = 1
# 0: False. 1: True. 2: False, but set direction of travel to stationary
@@ -1004,18 +1024,19 @@ async def _async_get_gps_accuracy(self):
proceed_with_update = 0
# 0: False. 1: True. 2: False, but set direction of travel to stationary
_LOGGER.info(
- f"({self._get_attr(CONF_NAME)}) GPS Accuracy is 0.0, not performing update"
+ "(%s) GPS Accuracy is 0.0, not performing update", self._get_attr(CONF_NAME)
)
else:
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) GPS Accuracy: "
- f"{round(self._get_attr(ATTR_GPS_ACCURACY), 3)}"
+ "(%s) GPS Accuracy: %s",
+ self._get_attr(CONF_NAME),
+ round(self._get_attr_safe_float(ATTR_GPS_ACCURACY), 3),
)
return proceed_with_update
- async def _async_get_driving_status(self):
+ async def _async_get_driving_status(self) -> None:
self._clear_attr(ATTR_DRIVING)
- isDriving = False
+ isDriving: bool = False
if not await self._async_in_zone():
if self._get_attr(ATTR_DIRECTION_OF_TRAVEL) != "stationary" and (
self._get_attr(ATTR_PLACE_CATEGORY) == "highway"
@@ -1025,243 +1046,239 @@ async def _async_get_driving_status(self):
if isDriving:
self._set_attr(ATTR_DRIVING, "Driving")
- async def _async_parse_osm_dict(self):
- if "type" in (self._get_attr(ATTR_OSM_DICT)):
- self._set_attr(ATTR_PLACE_TYPE, self._get_attr(ATTR_OSM_DICT).get("type"))
- if self._get_attr(ATTR_PLACE_TYPE) == "yes":
- if "addresstype" in (self._get_attr(ATTR_OSM_DICT)):
- self._set_attr(
- ATTR_PLACE_TYPE,
- self._get_attr(ATTR_OSM_DICT).get("addresstype"),
- )
- else:
- self._clear_attr(ATTR_PLACE_TYPE)
- if "address" in (self._get_attr(ATTR_OSM_DICT)) and self._get_attr(
- ATTR_PLACE_TYPE
- ) in (self._get_attr(ATTR_OSM_DICT)).get("address"):
+ async def _async_parse_osm_dict(self) -> None:
+ osm_dict: MutableMapping[str, Any] | None = self._get_attr(ATTR_OSM_DICT)
+ if not osm_dict:
+ return
+
+ await self._parse_type(osm_dict=osm_dict)
+ await self._parse_category(osm_dict=osm_dict)
+ await self._parse_namedetails(osm_dict=osm_dict)
+ await self._parse_address(osm_dict=osm_dict)
+ await self._parse_miscellaneous(osm_dict=osm_dict)
+ await self._set_place_name_no_dupe()
+
+ _LOGGER.debug(
+ "(%s) Entity attributes after parsing OSM Dict: %s",
+ self._get_attr(CONF_NAME),
+ self._internal_attr,
+ )
+
+ async def _parse_type(self, osm_dict: MutableMapping[str, Any]) -> None:
+ if "type" not in osm_dict:
+ return
+ self._set_attr(ATTR_PLACE_TYPE, osm_dict.get("type"))
+ if self._get_attr(ATTR_PLACE_TYPE) == "yes":
+ if "addresstype" in osm_dict:
self._set_attr(
- ATTR_PLACE_NAME,
- self._get_attr(ATTR_OSM_DICT)
- .get("address")
- .get(self._get_attr(ATTR_PLACE_TYPE)),
+ ATTR_PLACE_TYPE,
+ osm_dict.get("addresstype"),
)
- if "category" in (self._get_attr(ATTR_OSM_DICT)):
+ else:
+ self._clear_attr(ATTR_PLACE_TYPE)
+ if "address" in osm_dict and self._get_attr(ATTR_PLACE_TYPE) in osm_dict["address"]:
self._set_attr(
- ATTR_PLACE_CATEGORY,
- self._get_attr(ATTR_OSM_DICT).get("category"),
+ ATTR_PLACE_NAME,
+ osm_dict["address"].get(self._get_attr(ATTR_PLACE_TYPE)),
)
- if "address" in (self._get_attr(ATTR_OSM_DICT)) and self._get_attr(
- ATTR_PLACE_CATEGORY
- ) in (self._get_attr(ATTR_OSM_DICT)).get("address"):
- self._set_attr(
- ATTR_PLACE_NAME,
- (self._get_attr(ATTR_OSM_DICT))
- .get("address")
- .get(self._get_attr(ATTR_PLACE_CATEGORY)),
- )
- if (
- "namedetails" in (self._get_attr(ATTR_OSM_DICT))
- and self._get_attr(ATTR_OSM_DICT).get("namedetails") is not None
- ):
- if "name" in (self._get_attr(ATTR_OSM_DICT)).get("namedetails"):
- self._set_attr(
- ATTR_PLACE_NAME,
- (self._get_attr(ATTR_OSM_DICT)).get("namedetails").get("name"),
- )
- if not self._is_attr_blank(CONF_LANGUAGE):
- for language in (self._get_attr(CONF_LANGUAGE)).split(","):
- if "name:" + language in (self._get_attr(ATTR_OSM_DICT)).get(
- "namedetails"
- ):
- self._set_attr(
- ATTR_PLACE_NAME,
- self._get_attr(ATTR_OSM_DICT)
- .get("namedetails")
- .get("name:" + language),
- )
- break
- if (
- "address" in (self._get_attr(ATTR_OSM_DICT))
- and (self._get_attr(ATTR_OSM_DICT)).get("address") is not None
- ):
- if "house_number" in (self._get_attr(ATTR_OSM_DICT)).get("address"):
- self._set_attr(
- ATTR_STREET_NUMBER,
- (
- (self._get_attr(ATTR_OSM_DICT))
- .get("address")
- .get("house_number")
- ),
- )
- if "road" in (self._get_attr(ATTR_OSM_DICT)).get("address"):
- self._set_attr(
- ATTR_STREET,
- (self._get_attr(ATTR_OSM_DICT)).get("address").get("road"),
- )
- if "retail" in (self._get_attr(ATTR_OSM_DICT)).get("address") and (
- self._is_attr_blank(ATTR_PLACE_NAME)
- or (
- not self._is_attr_blank(ATTR_PLACE_CATEGORY)
- and not self._is_attr_blank(ATTR_STREET)
- and self._get_attr(ATTR_PLACE_CATEGORY) == "highway"
- and self._get_attr(ATTR_STREET) == self._get_attr(ATTR_PLACE_NAME)
- )
- ):
- self._set_attr(
- ATTR_PLACE_NAME,
- self._get_attr(ATTR_OSM_DICT).get("address").get("retail"),
- )
- _LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) Place Name: "
- f"{self._get_attr(ATTR_PLACE_NAME)}"
- )
-
- CITY_LIST = [
- "city",
- "town",
- "village",
- "township",
- "hamlet",
- "city_district",
- "municipality",
- ]
- POSTAL_TOWN_LIST = [
- "city",
- "town",
- "village",
- "township",
- "hamlet",
- "borough",
- "suburb",
- ]
- NEIGHBOURHOOD_LIST = [
- "village",
- "township",
- "hamlet",
- "borough",
- "suburb",
- "quarter",
- "neighbourhood",
- ]
- _LOGGER.debug(f"CITY_LIST: {CITY_LIST}")
- for city_type in CITY_LIST:
- try:
- POSTAL_TOWN_LIST.remove(city_type)
- except ValueError:
- pass
- try:
- NEIGHBOURHOOD_LIST.remove(city_type)
- except ValueError:
- pass
- if city_type in (self._get_attr(ATTR_OSM_DICT)).get("address"):
- self._set_attr(
- ATTR_CITY,
- (self._get_attr(ATTR_OSM_DICT)).get("address").get(city_type),
- )
- break
- _LOGGER.debug(f"POSTAL_TOWN_LIST: {POSTAL_TOWN_LIST}")
- for postal_town_type in POSTAL_TOWN_LIST:
- try:
- NEIGHBOURHOOD_LIST.remove(postal_town_type)
- except ValueError:
- pass
- if postal_town_type in (self._get_attr(ATTR_OSM_DICT)).get("address"):
- self._set_attr(
- ATTR_POSTAL_TOWN,
- (self._get_attr(ATTR_OSM_DICT))
- .get("address")
- .get(postal_town_type),
- )
- break
- _LOGGER.debug(f"NEIGHBOURHOOD_LIST: {NEIGHBOURHOOD_LIST}")
- for neighbourhood_type in NEIGHBOURHOOD_LIST:
- if neighbourhood_type in (self._get_attr(ATTR_OSM_DICT)).get("address"):
+ async def _parse_category(self, osm_dict: MutableMapping[str, Any]) -> None:
+ if "category" not in osm_dict:
+ return
+
+ self._set_attr(
+ ATTR_PLACE_CATEGORY,
+ osm_dict.get("category"),
+ )
+ if "address" in osm_dict and self._get_attr(ATTR_PLACE_CATEGORY) in osm_dict["address"]:
+ self._set_attr(
+ ATTR_PLACE_NAME,
+ osm_dict["address"].get(self._get_attr(ATTR_PLACE_CATEGORY)),
+ )
+
+ async def _parse_namedetails(self, osm_dict: MutableMapping[str, Any]) -> None:
+ namedetails: MutableMapping[str, Any] | None = osm_dict.get("namedetails")
+ if not namedetails:
+ return
+ if "name" in namedetails:
+ self._set_attr(
+ ATTR_PLACE_NAME,
+ namedetails.get("name"),
+ )
+ if not self._is_attr_blank(CONF_LANGUAGE):
+ for language in self._get_attr_safe_str(CONF_LANGUAGE).split(","):
+ if f"name:{language}" in namedetails:
self._set_attr(
- ATTR_PLACE_NEIGHBOURHOOD,
- (self._get_attr(ATTR_OSM_DICT))
- .get("address")
- .get(neighbourhood_type),
+ ATTR_PLACE_NAME,
+ namedetails.get(f"name:{language}"),
)
break
- if not self._is_attr_blank(ATTR_CITY):
- self._set_attr(
- ATTR_CITY_CLEAN,
- (self._get_attr(ATTR_CITY)).replace(" Township", "").strip(),
- )
- if (self._get_attr(ATTR_CITY_CLEAN)).startswith("City of"):
- self._set_attr(
- ATTR_CITY_CLEAN,
- (self._get_attr(ATTR_CITY_CLEAN))[8:] + " City",
- )
+ async def _parse_address(self, osm_dict: MutableMapping[str, Any]) -> None:
+ address: MutableMapping[str, Any] | None = osm_dict.get("address")
+ if not address:
+ return
- if "state" in (self._get_attr(ATTR_OSM_DICT)).get("address"):
- self._set_attr(
- ATTR_REGION,
- (self._get_attr(ATTR_OSM_DICT)).get("address").get("state"),
- )
- if "ISO3166-2-lvl4" in (self._get_attr(ATTR_OSM_DICT)).get("address"):
- self._set_attr(
- ATTR_STATE_ABBR,
- (
- (self._get_attr(ATTR_OSM_DICT))
- .get("address")
- .get("ISO3166-2-lvl4")
- .split("-")[1]
- .upper()
- ),
- )
- if "county" in (self._get_attr(ATTR_OSM_DICT)).get("address"):
+ await self._set_address_details(address)
+ await self._set_city_details(address)
+ await self._set_region_details(address)
+
+ async def _set_address_details(self, address: MutableMapping[str, Any]) -> None:
+ if "house_number" in address:
+ self._set_attr(
+ ATTR_STREET_NUMBER,
+ address.get("house_number"),
+ )
+ if "road" in address:
+ self._set_attr(
+ ATTR_STREET,
+ address.get("road"),
+ )
+ if "retail" in address and (
+ self._is_attr_blank(ATTR_PLACE_NAME)
+ or (
+ not self._is_attr_blank(ATTR_PLACE_CATEGORY)
+ and not self._is_attr_blank(ATTR_STREET)
+ and self._get_attr(ATTR_PLACE_CATEGORY) == "highway"
+ and self._get_attr(ATTR_STREET) == self._get_attr(ATTR_PLACE_NAME)
+ )
+ ):
+ self._set_attr(
+ ATTR_PLACE_NAME,
+ self._get_attr_safe_dict(ATTR_OSM_DICT).get("address", {}).get("retail"),
+ )
+ _LOGGER.debug(
+ "(%s) Place Name: %s", self._get_attr(CONF_NAME), self._get_attr(ATTR_PLACE_NAME)
+ )
+
+ async def _set_city_details(self, address: MutableMapping[str, Any]) -> None:
+ CITY_LIST: list[str] = [
+ "city",
+ "town",
+ "village",
+ "township",
+ "hamlet",
+ "city_district",
+ "municipality",
+ ]
+ POSTAL_TOWN_LIST: list[str] = [
+ "city",
+ "town",
+ "village",
+ "township",
+ "hamlet",
+ "borough",
+ "suburb",
+ ]
+ NEIGHBOURHOOD_LIST: list[str] = [
+ "village",
+ "township",
+ "hamlet",
+ "borough",
+ "suburb",
+ "quarter",
+ "neighbourhood",
+ ]
+ for city_type in CITY_LIST:
+ with contextlib.suppress(ValueError):
+ POSTAL_TOWN_LIST.remove(city_type)
+
+ with contextlib.suppress(ValueError):
+ NEIGHBOURHOOD_LIST.remove(city_type)
+ if city_type in address:
self._set_attr(
- ATTR_COUNTY,
- (self._get_attr(ATTR_OSM_DICT)).get("address").get("county"),
+ ATTR_CITY,
+ address.get(city_type),
)
- if "country" in (self._get_attr(ATTR_OSM_DICT)).get("address"):
+ break
+ for postal_town_type in POSTAL_TOWN_LIST:
+ with contextlib.suppress(ValueError):
+ NEIGHBOURHOOD_LIST.remove(postal_town_type)
+ if postal_town_type in address:
self._set_attr(
- ATTR_COUNTRY,
- (self._get_attr(ATTR_OSM_DICT)).get("address").get("country"),
+ ATTR_POSTAL_TOWN,
+ address.get(postal_town_type),
)
- if "country_code" in (self._get_attr(ATTR_OSM_DICT)).get("address"):
+ break
+ for neighbourhood_type in NEIGHBOURHOOD_LIST:
+ if neighbourhood_type in address:
self._set_attr(
- ATTR_COUNTRY_CODE,
- (self._get_attr(ATTR_OSM_DICT))
- .get("address")
- .get("country_code")
- .upper(),
+ ATTR_PLACE_NEIGHBOURHOOD,
+ address.get(neighbourhood_type),
)
- if "postcode" in (self._get_attr(ATTR_OSM_DICT)).get("address"):
+ break
+
+ if not self._is_attr_blank(ATTR_CITY):
+ self._set_attr(
+ ATTR_CITY_CLEAN,
+ self._get_attr_safe_str(ATTR_CITY).replace(" Township", "").strip(),
+ )
+ if self._get_attr_safe_str(ATTR_CITY_CLEAN).startswith("City of"):
self._set_attr(
- ATTR_POSTAL_CODE,
- self._get_attr(ATTR_OSM_DICT).get("address").get("postcode"),
+ ATTR_CITY_CLEAN,
+ f"{self._get_attr_safe_str(ATTR_CITY_CLEAN)[8:]} City",
)
- if "display_name" in (self._get_attr(ATTR_OSM_DICT)):
+
+ async def _set_region_details(self, address: MutableMapping[str, Any]) -> None:
+ if "state" in address:
+ self._set_attr(
+ ATTR_REGION,
+ address.get("state"),
+ )
+ if "ISO3166-2-lvl4" in address:
+ self._set_attr(
+ ATTR_STATE_ABBR,
+ address["ISO3166-2-lvl4"].split("-")[1].upper(),
+ )
+ if "county" in address:
+ self._set_attr(
+ ATTR_COUNTY,
+ address.get("county"),
+ )
+ if "country" in address:
+ self._set_attr(
+ ATTR_COUNTRY,
+ address.get("country"),
+ )
+ if "country_code" in address:
+ self._set_attr(
+ ATTR_COUNTRY_CODE,
+ address["country_code"].upper(),
+ )
+ if "postcode" in address:
+ self._set_attr(
+ ATTR_POSTAL_CODE,
+ self._get_attr_safe_dict(ATTR_OSM_DICT).get("address", {}).get("postcode"),
+ )
+
+ async def _parse_miscellaneous(self, osm_dict: MutableMapping[str, Any]) -> None:
+ if "display_name" in osm_dict:
self._set_attr(
ATTR_FORMATTED_ADDRESS,
- (self._get_attr(ATTR_OSM_DICT)).get("display_name"),
+ osm_dict.get("display_name"),
)
- if "osm_id" in (self._get_attr(ATTR_OSM_DICT)):
+ if "osm_id" in osm_dict:
self._set_attr(
ATTR_OSM_ID,
- str(self._get_attr(ATTR_OSM_DICT).get("osm_id")),
+ str(self._get_attr_safe_dict(ATTR_OSM_DICT).get("osm_id", "")),
)
- if "osm_type" in (self._get_attr(ATTR_OSM_DICT)):
+ if "osm_type" in osm_dict:
self._set_attr(
ATTR_OSM_TYPE,
- (self._get_attr(ATTR_OSM_DICT)).get("osm_type"),
+ osm_dict.get("osm_type"),
)
if (
not self._is_attr_blank(ATTR_PLACE_CATEGORY)
- and (self._get_attr(ATTR_PLACE_CATEGORY)).lower() == "highway"
- and "namedetails" in (self._get_attr(ATTR_OSM_DICT))
- and (self._get_attr(ATTR_OSM_DICT)).get("namedetails") is not None
- and "ref" in (self._get_attr(ATTR_OSM_DICT)).get("namedetails")
+ and self._get_attr_safe_str(ATTR_PLACE_CATEGORY).lower() == "highway"
+ and "namedetails" in osm_dict
+ and osm_dict.get("namedetails") is not None
+ and "ref" in osm_dict["namedetails"]
):
- street_refs = re.split(
+ street_refs: list = re.split(
r"[\;\\\/\,\.\:]",
- (self._get_attr(ATTR_OSM_DICT)).get("namedetails").get("ref"),
+ osm_dict["namedetails"].get("ref"),
)
street_refs = [i for i in street_refs if i.strip()] # Remove blank strings
# _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) Street Refs: {street_refs}")
@@ -1271,14 +1288,21 @@ async def _async_parse_osm_dict(self):
break
if not self._is_attr_blank(ATTR_STREET_REF):
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) Street: "
- f"{self._get_attr(ATTR_STREET)} / "
- f"Street Ref: {self._get_attr(ATTR_STREET_REF)}"
+ "(%s) Street: %s / Street Ref: %s",
+ self._get_attr(CONF_NAME),
+ self._get_attr(ATTR_STREET),
+ self._get_attr(ATTR_STREET_REF),
)
- dupe_attributes_check = []
- for attr in PLACE_NAME_DUPLICATE_LIST:
- if not self._is_attr_blank(attr):
- dupe_attributes_check.append(self._get_attr(attr))
+
+ async def _set_place_name_no_dupe(self) -> None:
+ dupe_attributes_check: list[str] = []
+ dupe_attributes_check.extend(
+ [
+ self._get_attr_safe_str(attr)
+ for attr in PLACE_NAME_DUPLICATE_LIST
+ if not self._is_attr_blank(attr)
+ ]
+ )
if (
not self._is_attr_blank(ATTR_PLACE_NAME)
and self._get_attr(ATTR_PLACE_NAME) not in dupe_attributes_check
@@ -1286,24 +1310,28 @@ async def _async_parse_osm_dict(self):
self._set_attr(ATTR_PLACE_NAME_NO_DUPE, self._get_attr(ATTR_PLACE_NAME))
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) "
- "Entity attributes after parsing OSM Dict: "
- f"{self._internal_attr}"
+ "(%s) Entity attributes after parsing OSM Dict: %s",
+ self._get_attr(CONF_NAME),
+ self._internal_attr,
)
- async def _async_build_formatted_place(self):
- formatted_place_array = []
+ async def _async_build_formatted_place(self) -> None:
+ formatted_place_array: list[str] = []
if not await self._async_in_zone():
if not self._is_attr_blank(ATTR_DRIVING) and "driving" in (
- self._get_attr(ATTR_DISPLAY_OPTIONS_LIST)
+ self._get_attr_safe_list(ATTR_DISPLAY_OPTIONS_LIST)
):
- formatted_place_array.append(self._get_attr(ATTR_DRIVING))
+ formatted_place_array.append(self._get_attr_safe_str(ATTR_DRIVING))
# Don't use place name if the same as another attributes
- use_place_name = True
- sensor_attributes_values = []
- for attr in PLACE_NAME_DUPLICATE_LIST:
- if not self._is_attr_blank(attr):
- sensor_attributes_values.append(self._get_attr(attr))
+ use_place_name: bool = True
+ sensor_attributes_values: list[str] = []
+ sensor_attributes_values.extend(
+ [
+ self._get_attr_safe_str(attr)
+ for attr in PLACE_NAME_DUPLICATE_LIST
+ if not self._is_attr_blank(attr)
+ ]
+ )
# if not self._is_attr_blank(ATTR_PLACE_NAME):
# _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) Duplicated List [Place Name: {self._get_attr(ATTR_PLACE_NAME)}]: {sensor_attributes_values}")
if self._is_attr_blank(ATTR_PLACE_NAME):
@@ -1312,17 +1340,15 @@ async def _async_build_formatted_place(self):
elif self._get_attr(ATTR_PLACE_NAME) in sensor_attributes_values:
# _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) Not Using Place Name: {self._get_attr(ATTR_PLACE_NAME)}")
use_place_name = False
- _LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) use_place_name: {use_place_name}"
- )
+ _LOGGER.debug("(%s) use_place_name: %s", self._get_attr(CONF_NAME), use_place_name)
if not use_place_name:
if (
not self._is_attr_blank(ATTR_PLACE_TYPE)
- and (self._get_attr(ATTR_PLACE_TYPE)).lower() != "unclassified"
- and (self._get_attr(ATTR_PLACE_CATEGORY)).lower() != "highway"
+ and self._get_attr_safe_str(ATTR_PLACE_TYPE).lower() != "unclassified"
+ and self._get_attr_safe_str(ATTR_PLACE_CATEGORY).lower() != "highway"
):
formatted_place_array.append(
- (self._get_attr(ATTR_PLACE_TYPE))
+ self._get_attr_safe_str(ATTR_PLACE_TYPE)
.title()
.replace("Proposed", "")
.replace("Construction", "")
@@ -1330,259 +1356,248 @@ async def _async_build_formatted_place(self):
)
elif (
not self._is_attr_blank(ATTR_PLACE_CATEGORY)
- and (self._get_attr(ATTR_PLACE_CATEGORY)).lower() != "highway"
+ and self._get_attr_safe_str(ATTR_PLACE_CATEGORY).lower() != "highway"
):
formatted_place_array.append(
- (self._get_attr(ATTR_PLACE_CATEGORY)).title().strip()
- )
- street = None
- if self._is_attr_blank(ATTR_STREET) and not self._is_attr_blank(
- ATTR_STREET_REF
- ):
- street = (self._get_attr(ATTR_STREET_REF)).strip()
- _LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) Using street_ref: {street}"
+ self._get_attr_safe_str(ATTR_PLACE_CATEGORY).title().strip()
)
+ street: str | None = None
+ if self._is_attr_blank(ATTR_STREET) and not self._is_attr_blank(ATTR_STREET_REF):
+ street = self._get_attr_safe_str(ATTR_STREET_REF).strip()
+ _LOGGER.debug("(%s) Using street_ref: %s", self._get_attr(CONF_NAME), street)
elif not self._is_attr_blank(ATTR_STREET):
if (
not self._is_attr_blank(ATTR_PLACE_CATEGORY)
- and (self._get_attr(ATTR_PLACE_CATEGORY)).lower() == "highway"
+ and self._get_attr_safe_str(ATTR_PLACE_CATEGORY).lower() == "highway"
and not self._is_attr_blank(ATTR_PLACE_TYPE)
- and (self._get_attr(ATTR_PLACE_TYPE)).lower()
- in ["motorway", "trunk"]
+ and self._get_attr_safe_str(ATTR_PLACE_TYPE).lower()
+ in {"motorway", "trunk"}
and not self._is_attr_blank(ATTR_STREET_REF)
):
- street = (self._get_attr(ATTR_STREET_REF)).strip()
+ street = self._get_attr_safe_str(ATTR_STREET_REF).strip()
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) Using street_ref: {street}"
+ "(%s) Using street_ref: %s", self._get_attr(CONF_NAME), street
)
else:
- street = (self._get_attr(ATTR_STREET)).strip()
- _LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) Using street: {street}"
- )
+ street = self._get_attr_safe_str(ATTR_STREET).strip()
+ _LOGGER.debug("(%s) Using street: %s", self._get_attr(CONF_NAME), street)
if street and self._is_attr_blank(ATTR_STREET_NUMBER):
formatted_place_array.append(street)
elif street and not self._is_attr_blank(ATTR_STREET_NUMBER):
formatted_place_array.append(
- f"{str(self._get_attr(ATTR_STREET_NUMBER)).strip()} {street}"
+ f"{self._get_attr_safe_str(ATTR_STREET_NUMBER).strip()} {street}"
)
if (
not self._is_attr_blank(ATTR_PLACE_TYPE)
- and (self._get_attr(ATTR_PLACE_TYPE)).lower() == "house"
+ and self._get_attr_safe_str(ATTR_PLACE_TYPE).lower() == "house"
and not self._is_attr_blank(ATTR_PLACE_NEIGHBOURHOOD)
):
formatted_place_array.append(
- (self._get_attr(ATTR_PLACE_NEIGHBOURHOOD)).strip()
+ self._get_attr_safe_str(ATTR_PLACE_NEIGHBOURHOOD).strip()
)
else:
- formatted_place_array.append((self._get_attr(ATTR_PLACE_NAME)).strip())
+ formatted_place_array.append(self._get_attr_safe_str(ATTR_PLACE_NAME).strip())
if not self._is_attr_blank(ATTR_CITY_CLEAN):
- formatted_place_array.append((self._get_attr(ATTR_CITY_CLEAN)).strip())
+ formatted_place_array.append(self._get_attr_safe_str(ATTR_CITY_CLEAN).strip())
elif not self._is_attr_blank(ATTR_CITY):
- formatted_place_array.append((self._get_attr(ATTR_CITY)).strip())
+ formatted_place_array.append(self._get_attr_safe_str(ATTR_CITY).strip())
elif not self._is_attr_blank(ATTR_COUNTY):
- formatted_place_array.append((self._get_attr(ATTR_COUNTY)).strip())
+ formatted_place_array.append(self._get_attr_safe_str(ATTR_COUNTY).strip())
if not self._is_attr_blank(ATTR_STATE_ABBR):
- formatted_place_array.append(self._get_attr(ATTR_STATE_ABBR))
+ formatted_place_array.append(self._get_attr_safe_str(ATTR_STATE_ABBR))
else:
formatted_place_array.append(
- (self._get_attr(ATTR_DEVICETRACKER_ZONE_NAME)).strip()
+ self._get_attr_safe_str(ATTR_DEVICETRACKER_ZONE_NAME).strip()
)
- formatted_place = ", ".join(item for item in formatted_place_array)
+ formatted_place: str = ", ".join(item for item in formatted_place_array)
formatted_place = formatted_place.replace("\n", " ").replace(" ", " ").strip()
self._set_attr(ATTR_FORMATTED_PLACE, formatted_place)
- async def _async_build_from_advanced_options(self, curr_options):
- # _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) [adv_options] Options: {curr_options}")
+ async def _do_brackets_and_parens_count_match(self, curr_options: str) -> bool:
if curr_options.count("[") != curr_options.count("]"):
_LOGGER.error(
- f"({self._get_attr(CONF_NAME)}) "
- "[adv_options] Bracket Count Mismatch: "
- f"{curr_options}"
+ "(%s) [adv_options] Bracket Count Mismatch: %s",
+ self._get_attr(CONF_NAME),
+ curr_options,
)
- return
- elif curr_options.count("(") != curr_options.count(")"):
+ return False
+ if curr_options.count("(") != curr_options.count(")"):
_LOGGER.error(
- f"({self._get_attr(CONF_NAME)}) "
- "[adv_options] Parenthesis Count Mismatch: "
- f"{curr_options}"
+ "(%s) [adv_options] Parenthesis Count Mismatch: %s",
+ self._get_attr(CONF_NAME),
+ curr_options,
)
+ return False
+ return True
+
+ async def _async_build_from_advanced_options(self, curr_options: str):
+ _LOGGER.debug("(%s) [adv_options] Options: %s", self._get_attr(CONF_NAME), curr_options)
+ if not await self._do_brackets_and_parens_count_match(curr_options) or not curr_options:
return
- incl = []
- excl = []
- incl_attr = {}
- excl_attr = {}
- none_opt = None
- next_opt = None
- if curr_options is None or not curr_options:
+
+ # _LOGGER.debug("(%s) [adv_options] Options has a [ or ( and optional ,", self._get_attr(CONF_NAME))
+ if "[" in curr_options or "(" in curr_options:
+ await self._process_advanced_bracket_or_parens(curr_options=curr_options)
return
- elif "[" in curr_options or "(" in curr_options:
- # _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) [adv_options] Options has a [ or ( and optional ,")
- comma_num = curr_options.find(",")
- bracket_num = curr_options.find("[")
- paren_num = curr_options.find("(")
- if (
- comma_num != -1
- and (bracket_num == -1 or comma_num < bracket_num)
- and (paren_num == -1 or comma_num < paren_num)
- ):
- # Comma is first symbol
- # _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) [adv_options] Comma is First")
- opt = curr_options[:comma_num]
- # _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) [adv_options] Option: {opt}")
- if opt is not None and opt:
- ret_state = await self._async_get_option_state(opt.strip())
- if ret_state is not None and ret_state:
- self._adv_options_state_list.append(ret_state)
- _LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) "
- "[adv_options] Updated state list: "
- f"{self._adv_options_state_list}"
- )
- next_opt = curr_options[(comma_num + 1):]
- # _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) [adv_options] Next Options: {next_opt}")
- if next_opt is not None and next_opt:
- await self._async_build_from_advanced_options(next_opt.strip())
- # _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) [adv_options] Back from recursion")
- return
- elif (
- bracket_num != -1
- and (comma_num == -1 or bracket_num < comma_num)
- and (paren_num == -1 or bracket_num < paren_num)
- ):
- # Bracket is first symbol
- # _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) [adv_options] Bracket is First")
- opt = curr_options[:bracket_num]
- # _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) [adv_options] Option: {opt}")
- none_opt, next_opt = await self._async_parse_bracket(
- curr_options[bracket_num:]
- )
- if (
- next_opt is not None
- and next_opt
- and len(next_opt) > 1
- and next_opt[0] == "("
- ):
- # Parse Parenthesis
- incl, excl, incl_attr, excl_attr, next_opt = (
- await self._async_parse_parens(next_opt)
- )
- if opt is not None and opt:
- ret_state = await self._async_get_option_state(
- opt.strip(), incl, excl, incl_attr, excl_attr
+ # _LOGGER.debug("(%s) [adv_options] Options has , but no [ or (, splitting", self._get_attr(CONF_NAME))
+ if "," in curr_options:
+ await self._process_advanced_only_commas(curr_options=curr_options)
+ return
+
+ # _LOGGER.debug("(%s) [adv_options] Options should just be a single term", self._get_attr(CONF_NAME))
+ await self._process_advanced_single_term(curr_options=curr_options)
+
+ async def _process_advanced_bracket_or_parens(self, curr_options: str) -> None:
+ incl: list[str] = []
+ excl: list[str] = []
+ incl_attr: MutableMapping[str, Any] = {}
+ excl_attr: MutableMapping[str, Any] = {}
+ none_opt: str | None = None
+ next_opt: str | None = None
+
+ # _LOGGER.debug("(%s) [adv_options] Options has a [ or ( and optional ,", self._get_attr(CONF_NAME))
+ comma_num: int = curr_options.find(",")
+ bracket_num: int = curr_options.find("[")
+ paren_num: int = curr_options.find("(")
+
+ # Comma is first symbol
+ if (
+ comma_num != -1
+ and (bracket_num == -1 or comma_num < bracket_num)
+ and (paren_num == -1 or comma_num < paren_num)
+ ):
+ # _LOGGER.debug("(%s) [adv_options] Comma is First", self._get_attr(CONF_NAME))
+ opt: str = curr_options[:comma_num]
+ # _LOGGER.debug("(%s) [adv_options] Option: %s", self._get_attr(CONF_NAME), opt)
+ if opt:
+ ret_state: str | None = await self._async_get_option_state(opt.strip())
+ if ret_state:
+ self._adv_options_state_list.append(ret_state)
+ _LOGGER.debug(
+ "(%s) [adv_options] Updated state list: %s",
+ self._get_attr(CONF_NAME),
+ self._adv_options_state_list,
)
- if ret_state is not None and ret_state:
- self._adv_options_state_list.append(ret_state)
- _LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) "
- "[adv_options] Updated state list: "
- f"{self._adv_options_state_list}"
- )
- elif none_opt is not None and none_opt:
- await self._async_build_from_advanced_options(none_opt.strip())
- # _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) [adv_options] Back from recursion")
+ next_opt = curr_options[(comma_num + 1) :]
+ # _LOGGER.debug("(%s) [adv_options] Next Options: %s",self._get_attr(CONF_NAME), next_opt)
+ if next_opt:
+ await self._async_build_from_advanced_options(next_opt.strip())
+ # _LOGGER.debug("(%s) [adv_options] Back from recursion", self._get_attr(CONF_NAME))
+ return
- if (
- next_opt is not None
- and next_opt
- and len(next_opt) > 1
- and next_opt[0] == ","
- ):
- next_opt = next_opt[1:]
- # _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) [adv_options] Next Options: {next_opt}")
- if next_opt is not None and next_opt:
- await self._async_build_from_advanced_options(next_opt.strip())
- # _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) [adv_options] Back from recursion")
- return
- elif (
- paren_num != -1
- and (comma_num == -1 or paren_num < comma_num)
- and (bracket_num == -1 or paren_num < bracket_num)
- ):
- # Parenthesis is first symbol
- # _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) [adv_options] Parenthesis is First")
- opt = curr_options[:paren_num]
- # _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) [adv_options] Option: {opt}")
- incl, excl, incl_attr, excl_attr, next_opt = (
- await self._async_parse_parens(curr_options[paren_num:])
+ # Bracket is first symbol
+ if (
+ bracket_num != -1
+ and (comma_num == -1 or bracket_num < comma_num)
+ and (paren_num == -1 or bracket_num < paren_num)
+ ):
+ # _LOGGER.debug("(%s) [adv_options] Bracket is First", self._get_attr(CONF_NAME))
+ opt = curr_options[:bracket_num]
+ # _LOGGER.debug("(%s) [adv_options] Option: %s", self._get_attr(CONF_NAME), opt)
+ none_opt, next_opt = await self._async_parse_bracket(curr_options[bracket_num:])
+ if next_opt and len(next_opt) > 1 and next_opt[0] == "(":
+ # Parse Parenthesis
+ incl, excl, incl_attr, excl_attr, next_opt = await self._async_parse_parens(
+ next_opt
)
- if (
- next_opt is not None
- and next_opt
- and len(next_opt) > 1
- and next_opt[0] == "["
- ):
- # Parse Bracket
- none_opt, next_opt = await self._async_parse_bracket(next_opt)
- if opt is not None and opt:
- ret_state = await self._async_get_option_state(
- opt.strip(), incl, excl, incl_attr, excl_attr
+ if opt:
+ ret_state = await self._async_get_option_state(
+ opt.strip(), incl, excl, incl_attr, excl_attr
+ )
+ if ret_state:
+ self._adv_options_state_list.append(ret_state)
+ _LOGGER.debug(
+ "(%s) [adv_options] Updated state list: %s",
+ self._get_attr(CONF_NAME),
+ self._adv_options_state_list,
)
- if ret_state is not None and ret_state:
- self._adv_options_state_list.append(ret_state)
- _LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) "
- "[adv_options] Updated state list: "
- f"{self._adv_options_state_list}"
- )
- elif none_opt is not None and none_opt:
- await self._async_build_from_advanced_options(none_opt.strip())
- # _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) [adv_options] Back from recursion")
-
- if (
- next_opt is not None
- and next_opt
- and len(next_opt) > 1
- and next_opt[0] == ","
- ):
- next_opt = next_opt[1:]
- # _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) [adv_options] Next Options: {next_opt}")
- if next_opt is not None and next_opt:
- await self._async_build_from_advanced_options(next_opt.strip())
- # _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) [adv_options] Back from recursion")
- return
- return
- elif "," in curr_options:
- # _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) [adv_options] Options has , but no [ or (, splitting")
- for opt in curr_options.split(","):
- if opt is not None and opt:
- ret_state = await self._async_get_option_state(opt.strip())
- if ret_state is not None and ret_state:
- self._adv_options_state_list.append(ret_state)
- _LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) "
- "[adv_options] Updated state list: "
- f"{self._adv_options_state_list}"
- )
+ elif none_opt:
+ await self._async_build_from_advanced_options(none_opt.strip())
+ # _LOGGER.debug("(%s) [adv_options] Back from recursion", self._get_attr(CONF_NAME))
+
+ if next_opt and len(next_opt) > 1 and next_opt[0] == ",":
+ next_opt = next_opt[1:]
+ # _LOGGER.debug("(%s) [adv_options] Next Options: %s", self._get_attr(CONF_NAME), next_opt)
+ if next_opt:
+ await self._async_build_from_advanced_options(next_opt.strip())
+ # _LOGGER.debug("(%s) [adv_options] Back from recursion", self._get_attr(CONF_NAME))
return
- else:
- # _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) [adv_options] Options should just be a single term")
- ret_state = await self._async_get_option_state(curr_options.strip())
- if ret_state is not None and ret_state:
- self._adv_options_state_list.append(ret_state)
- _LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) "
- "[adv_options] Updated state list: "
- f"{self._adv_options_state_list}"
+
+ # Parenthesis is first symbol
+ if (
+ paren_num != -1
+ and (comma_num == -1 or paren_num < comma_num)
+ and (bracket_num == -1 or paren_num < bracket_num)
+ ):
+ # _LOGGER.debug("(%s) [adv_options] Parenthesis is First", self._get_attr(CONF_NAME))
+ opt = curr_options[:paren_num]
+ _LOGGER.debug("(%s) [adv_options] Option: %s", self._get_attr(CONF_NAME), opt)
+ incl, excl, incl_attr, excl_attr, next_opt = await self._async_parse_parens(
+ curr_options[paren_num:]
+ )
+ if next_opt and len(next_opt) > 1 and next_opt[0] == "[":
+ # Parse Bracket
+ none_opt, next_opt = await self._async_parse_bracket(next_opt)
+
+ if opt:
+ ret_state = await self._async_get_option_state(
+ opt.strip(), incl, excl, incl_attr, excl_attr
)
- return
- return
+ if ret_state:
+ self._adv_options_state_list.append(ret_state)
+ _LOGGER.debug(
+ "(%s) [adv_options] Updated state list: %s",
+ self._get_attr(CONF_NAME),
+ self._adv_options_state_list,
+ )
+ elif none_opt:
+ await self._async_build_from_advanced_options(none_opt.strip())
+ # _LOGGER.debug("(%s) [adv_options] Back from recursion", self._get_attr(CONF_NAME))
+
+ if next_opt and len(next_opt) > 1 and next_opt[0] == ",":
+ next_opt = next_opt[1:]
+ # _LOGGER.debug("(%s) [adv_options] Next Options: %s", self._get_attr(CONF_NAME), next_opt)
+ if next_opt:
+ await self._async_build_from_advanced_options(next_opt.strip())
+ # _LOGGER.debug("(%s) [adv_options] Back from recursion", self._get_attr(CONF_NAME))
+
+ async def _process_advanced_only_commas(self, curr_options: str) -> None:
+ # _LOGGER.debug("(%s) [adv_options] Options has , but no [ or (, splitting", self._get_attr(CONF_NAME))
+ for opt in curr_options.split(","):
+ if opt is not None and opt:
+ ret_state = await self._async_get_option_state(opt.strip())
+ if ret_state is not None and ret_state:
+ self._adv_options_state_list.append(ret_state)
+ _LOGGER.debug(
+ "(%s) [adv_options] Updated state list: %s",
+ self._get_attr(CONF_NAME),
+ self._adv_options_state_list,
+ )
+
+ async def _process_advanced_single_term(self, curr_options: str) -> None:
+ ret_state = await self._async_get_option_state(curr_options.strip())
+ if ret_state is not None and ret_state:
+ self._adv_options_state_list.append(ret_state)
+ _LOGGER.debug(
+ "(%s) [adv_options] Updated state list: %s",
+ self._get_attr(CONF_NAME),
+ self._adv_options_state_list,
+ )
- async def _async_parse_parens(self, curr_options):
- incl = []
- excl = []
- incl_attr = {}
- excl_attr = {}
- incl_excl_list = []
- empty_paren = False
+ async def _async_parse_parens(self, curr_options: str):
+ incl: list = []
+ excl: list = []
+ incl_attr: MutableMapping[str, Any] = {}
+ excl_attr: MutableMapping[str, Any] = {}
+ incl_excl_list: list = []
+ empty_paren: bool = False
next_opt = None
- paren_count = 1
- close_paren_num = 0
- last_comma = -1
+ paren_count: int = 1
+ close_paren_num: int = 0
+ last_comma: int = -1
if curr_options[0] == "(":
curr_options = curr_options[1:]
if curr_options[0] == ")":
@@ -1590,8 +1605,8 @@ async def _async_parse_parens(self, curr_options):
close_paren_num = 0
else:
for i, c in enumerate(curr_options):
- if c in [",", ")"] and paren_count == 1:
- incl_excl_list.append(curr_options[(last_comma + 1): i].strip())
+ if c in {",", ")"} and paren_count == 1:
+ incl_excl_list.append(curr_options[(last_comma + 1) : i].strip())
last_comma = i
if c == "(":
paren_count += 1
@@ -1603,8 +1618,8 @@ async def _async_parse_parens(self, curr_options):
if close_paren_num > 0 and paren_count == 0 and incl_excl_list:
# _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) [parse_parens] incl_excl_list: {incl_excl_list}")
- paren_first = True
- paren_incl = True
+ paren_first: bool = True
+ paren_incl: bool = True
for item in incl_excl_list:
if paren_first:
paren_first = False
@@ -1619,23 +1634,18 @@ async def _async_parse_parens(self, curr_options):
# _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) [parse_parens] item: {item}")
if item is not None and item:
if "(" in item:
- if (
- ")" not in item
- or item.count("(") > 1
- or item.count(")") > 1
- ):
+ if ")" not in item or item.count("(") > 1 or item.count(")") > 1:
_LOGGER.error(
- f"({self._get_attr(CONF_NAME)}) "
- f"[parse_parens] Parenthesis Mismatch: {item}"
+ "(%s) [parse_parens] Parenthesis Mismatch: %s",
+ self._get_attr(CONF_NAME),
+ item,
)
continue
paren_attr = item[: item.find("(")]
paren_attr_first = True
paren_attr_incl = True
- paren_attr_list = []
- for attr_item in item[
- (item.find("(") + 1): item.find(")")
- ].split(","):
+ paren_attr_list: list = []
+ for attr_item in item[(item.find("(") + 1) : item.find(")")].split(","):
if paren_attr_first:
paren_attr_first = False
if attr_item == "-":
@@ -1659,20 +1669,21 @@ async def _async_parse_parens(self, curr_options):
elif not empty_paren:
_LOGGER.error(
- f"({self._get_attr(CONF_NAME)}) "
- f"[parse_parens] Parenthesis Mismatch: {curr_options}"
+ "(%s) [parse_parens] Parenthesis Mismatch: %s",
+ self._get_attr(CONF_NAME),
+ curr_options,
)
- next_opt = curr_options[(close_paren_num + 1):]
+ next_opt = curr_options[(close_paren_num + 1) :]
# _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) [parse_parens] Raw Next Options: {next_opt}")
return incl, excl, incl_attr, excl_attr, next_opt
- async def _async_parse_bracket(self, curr_options):
+ async def _async_parse_bracket(self, curr_options: str):
# _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) [parse_bracket] Options: {curr_options}")
- empty_bracket = False
- none_opt = None
- next_opt = None
- bracket_count = 1
- close_bracket_num = 0
+ empty_bracket: bool = False
+ none_opt: str | None = None
+ next_opt: str | None = None
+ bracket_count: int = 1
+ close_bracket_num: int = 0
if curr_options[0] == "[":
curr_options = curr_options[1:]
if curr_options[0] == "]":
@@ -1692,62 +1703,89 @@ async def _async_parse_bracket(self, curr_options):
if empty_bracket or (close_bracket_num > 0 and bracket_count == 0):
none_opt = curr_options[:close_bracket_num].strip()
# _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) [parse_bracket] None Options: {none_opt}")
- next_opt = curr_options[(close_bracket_num + 1):].strip()
+ next_opt = curr_options[(close_bracket_num + 1) :].strip()
# _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) [parse_bracket] Raw Next Options: {next_opt}")
else:
_LOGGER.error(
- f"({self._get_attr(CONF_NAME)}) "
- f"[parse_bracket] Bracket Mismatch Error: {curr_options}"
+ "(%s) [parse_bracket] Bracket Mismatch Error: %s",
+ self._get_attr(CONF_NAME),
+ curr_options,
)
return none_opt, next_opt
async def _async_get_option_state(
- self, opt, incl=None, excl=None, incl_attr=None, excl_attr=None
- ):
+ self,
+ opt: str,
+ incl: list | None = None,
+ excl: list | None = None,
+ incl_attr: MutableMapping[str, Any] | None = None,
+ excl_attr: MutableMapping[str, Any] | None = None,
+ ) -> str | None:
incl = [] if incl is None else incl
excl = [] if excl is None else excl
incl_attr = {} if incl_attr is None else incl_attr
excl_attr = {} if excl_attr is None else excl_attr
- if opt is not None and opt:
+ if opt:
opt = str(opt).lower().strip()
- # _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) [get_option_state] Option: {opt}")
- out = self._get_attr(DISPLAY_OPTIONS_MAP.get(opt))
+ _LOGGER.debug("(%s) [get_option_state] Option: %s", self._get_attr(CONF_NAME), opt)
+ out: str | None = self._get_attr(DISPLAY_OPTIONS_MAP.get(opt))
if (
- DISPLAY_OPTIONS_MAP.get(opt)
- in [ATTR_DEVICETRACKER_ZONE, ATTR_DEVICETRACKER_ZONE_NAME]
+ DISPLAY_OPTIONS_MAP.get(opt) in {ATTR_DEVICETRACKER_ZONE, ATTR_DEVICETRACKER_ZONE_NAME}
and not await self._async_in_zone()
):
out = None
- _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) [get_option_state] State: {out}")
- # _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) [get_option_state] incl list: {incl}")
- # _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) [get_option_state] excl list: {excl}")
- # _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) [get_option_state] incl_attr dict: {incl_attr}")
- # _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) [get_option_state] excl_attr dict: {excl_attr}")
- if out is not None and out:
- if incl and str(out).strip().lower() not in incl:
- out = None
- elif excl and str(out).strip().lower() in excl:
+ _LOGGER.debug("(%s) [get_option_state] State: %s", self._get_attr(CONF_NAME), out)
+ _LOGGER.debug("(%s) [get_option_state] incl list: %s", self._get_attr(CONF_NAME), incl)
+ _LOGGER.debug("(%s) [get_option_state] excl list: %s", self._get_attr(CONF_NAME), excl)
+ _LOGGER.debug(
+ "(%s) [get_option_state] incl_attr dict: %s", self._get_attr(CONF_NAME), incl_attr
+ )
+ _LOGGER.debug(
+ "(%s) [get_option_state] excl_attr dict: %s", self._get_attr(CONF_NAME), excl_attr
+ )
+ if out:
+ if (
+ incl
+ and str(out).strip().lower() not in incl
+ or excl
+ and str(out).strip().lower() in excl
+ ):
out = None
if incl_attr:
for attr, states in incl_attr.items():
- # _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) [get_option_state] incl_attr: {attr} / State: {self._get_attr(DISPLAY_OPTIONS_MAP.get(attr))}")
- # _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) [get_option_state] incl_states: {states}")
+ _LOGGER.debug(
+ "(%s) [get_option_state] incl_attr: %s / State: %s",
+ self._get_attr(CONF_NAME),
+ attr,
+ self._get_attr(DISPLAY_OPTIONS_MAP.get(attr)),
+ )
+ _LOGGER.debug(
+ "(%s) [get_option_state] incl_states: %s", self._get_attr(CONF_NAME), states
+ )
+ map_attr: str | None = DISPLAY_OPTIONS_MAP.get(attr)
if (
- self._is_attr_blank(DISPLAY_OPTIONS_MAP.get(attr))
- or self._get_attr(DISPLAY_OPTIONS_MAP.get(attr)) not in states
+ not map_attr
+ or self._is_attr_blank(map_attr)
+ or self._get_attr(map_attr) not in states
):
out = None
if excl_attr:
for attr, states in excl_attr.items():
- # _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) [get_option_state] excl_attr: {attr} / State: {self._get_attr(DISPLAY_OPTIONS_MAP.get(attr))}")
- # _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) [get_option_state] excl_states: {states}")
+ _LOGGER.debug(
+ "(%s) [get_option_state] excl_attr: %s / State: %s",
+ self._get_attr(CONF_NAME),
+ attr,
+ self._get_attr(DISPLAY_OPTIONS_MAP.get(attr)),
+ )
+ _LOGGER.debug(
+ "(%s) [get_option_state] excl_states: %s", self._get_attr(CONF_NAME), states
+ )
if self._get_attr(DISPLAY_OPTIONS_MAP.get(attr)) in states:
out = None
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) "
- f"[get_option_state] State after incl/excl: {out}"
+ "(%s) [get_option_state] State after incl/excl: %s", self._get_attr(CONF_NAME), out
)
- if out is not None and out:
+ if out:
if out == out.lower() and (
DISPLAY_OPTIONS_MAP.get(opt) == ATTR_DEVICETRACKER_ZONE_NAME
or DISPLAY_OPTIONS_MAP.get(opt) == ATTR_PLACE_TYPE
@@ -1766,12 +1804,11 @@ async def _async_get_option_state(
# _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) [get_option_state] street_num_i: {self._street_num_i}")
self._temp_i += 1
return out
- else:
- return None
+ return None
- async def _async_compile_state_from_advanced_options(self):
+ async def _async_compile_state_from_advanced_options(self) -> None:
self._street_num_i += 1
- first = True
+ first: bool = True
for i, out in enumerate(self._adv_options_state_list):
if out is not None and out:
out = out.strip()
@@ -1795,128 +1832,115 @@ async def _async_compile_state_from_advanced_options(self):
)
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) New State from Advanced Display Options: "
- f"{self._get_attr(ATTR_NATIVE_VALUE)}"
+ "(%s) New State from Advanced Display Options: %s",
+ self._get_attr(CONF_NAME),
+ self._get_attr(ATTR_NATIVE_VALUE),
)
- async def _async_build_state_from_display_options(self):
- # Options: "formatted_place, driving, zone, zone_name, place_name, place, street_number, street, city, county, state, postal_code, country, formatted_address, do_not_show_not_home"
-
- display_options = self._get_attr(ATTR_DISPLAY_OPTIONS_LIST)
+ async def _async_build_state_from_display_options(self) -> None:
+ display_options = self._get_attr_safe_list(ATTR_DISPLAY_OPTIONS_LIST)
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) Building State from Display Options: "
- f"{self._get_attr(ATTR_DISPLAY_OPTIONS)}"
+ "(%s) Building State from Display Options: %s",
+ self._get_attr(CONF_NAME),
+ self._get_attr(ATTR_DISPLAY_OPTIONS),
)
- user_display = []
- if "driving" in display_options and not self._is_attr_blank(ATTR_DRIVING):
- user_display.append(self._get_attr(ATTR_DRIVING))
+ def add_to_display(option_key: str, attr_key: str, condition: bool = True) -> None:
+ """Add attribute value to user_display if the conditions are met."""
+ if option_key in display_options and not self._is_attr_blank(attr_key) and condition:
+ user_display.append(self._get_attr_safe_str(attr_key))
- if (
- "zone_name" in display_options
- and not self._is_attr_blank(ATTR_DEVICETRACKER_ZONE_NAME)
- and (
- await self._async_in_zone()
- or "do_not_show_not_home" not in display_options
- )
- ):
- user_display.append(self._get_attr(ATTR_DEVICETRACKER_ZONE_NAME))
- elif (
- "zone" in display_options
- and not self._is_attr_blank(ATTR_DEVICETRACKER_ZONE)
- and (
- await self._async_in_zone()
- or "do_not_show_not_home" not in display_options
- )
- ):
- user_display.append(self._get_attr(ATTR_DEVICETRACKER_ZONE))
+ user_display: list[str] = []
- if "place_name" in display_options and not self._is_attr_blank(ATTR_PLACE_NAME):
- user_display.append(self._get_attr(ATTR_PLACE_NAME))
+ # Add basic options
+ add_to_display("driving", ATTR_DRIVING)
+ add_to_display(
+ "zone_name",
+ ATTR_DEVICETRACKER_ZONE_NAME,
+ await self._async_in_zone() or "do_not_show_not_home" not in display_options,
+ )
+ add_to_display(
+ "zone",
+ ATTR_DEVICETRACKER_ZONE,
+ await self._async_in_zone() or "do_not_show_not_home" not in display_options,
+ )
+ add_to_display("place_name", ATTR_PLACE_NAME)
+
+ # Handle "place" and its sub-options
if "place" in display_options:
if not self._is_attr_blank(ATTR_PLACE_NAME) and self._get_attr(
ATTR_PLACE_NAME
) != self._get_attr(ATTR_STREET):
- user_display.append(self._get_attr(ATTR_PLACE_NAME))
- if (
- not self._is_attr_blank(ATTR_PLACE_CATEGORY)
- and (self._get_attr(ATTR_PLACE_CATEGORY)).lower() != "place"
- ):
- user_display.append(self._get_attr(ATTR_PLACE_CATEGORY))
- if (
- not self._is_attr_blank(ATTR_PLACE_TYPE)
- and (self._get_attr(ATTR_PLACE_TYPE)).lower() != "yes"
- ):
- user_display.append(self._get_attr(ATTR_PLACE_TYPE))
- if not self._is_attr_blank(ATTR_PLACE_NEIGHBOURHOOD):
- user_display.append(self._get_attr(ATTR_PLACE_NEIGHBOURHOOD))
- if not self._is_attr_blank(ATTR_STREET_NUMBER):
- user_display.append(self._get_attr(ATTR_STREET_NUMBER))
- if not self._is_attr_blank(ATTR_STREET):
- user_display.append(self._get_attr(ATTR_STREET))
+ user_display.append(self._get_attr_safe_str(ATTR_PLACE_NAME))
+ for key, attr in {
+ "place_category": ATTR_PLACE_CATEGORY,
+ "place_type": ATTR_PLACE_TYPE,
+ "place_neighbourhood": ATTR_PLACE_NEIGHBOURHOOD,
+ "street_number": ATTR_STREET_NUMBER,
+ "street": ATTR_STREET,
+ }.items():
+ add_to_display(
+ key, attr, (self._get_attr(attr) or "").lower() not in {"place", "yes"}
+ )
else:
- if "street_number" in display_options and not self._is_attr_blank(
- ATTR_STREET_NUMBER
- ):
- user_display.append(self._get_attr(ATTR_STREET_NUMBER))
- if "street" in display_options and not self._is_attr_blank(ATTR_STREET):
- user_display.append(self._get_attr(ATTR_STREET))
- if "city" in display_options and not self._is_attr_blank(ATTR_CITY):
- user_display.append(self._get_attr(ATTR_CITY))
- if "county" in display_options and not self._is_attr_blank(ATTR_COUNTY):
- user_display.append(self._get_attr(ATTR_COUNTY))
- if "state" in display_options and not self._is_attr_blank(ATTR_REGION):
- user_display.append(self._get_attr(ATTR_REGION))
- elif "region" in display_options and not self._is_attr_blank(ATTR_REGION):
- user_display.append(self._get_attr(ATTR_REGION))
- if "postal_code" in display_options and not self._is_attr_blank(
- ATTR_POSTAL_CODE
- ):
- user_display.append(self._get_attr(ATTR_POSTAL_CODE))
- if "country" in display_options and not self._is_attr_blank(ATTR_COUNTRY):
- user_display.append(self._get_attr(ATTR_COUNTRY))
- if "formatted_address" in display_options and not self._is_attr_blank(
- ATTR_FORMATTED_ADDRESS
- ):
- user_display.append(self._get_attr(ATTR_FORMATTED_ADDRESS))
-
+ add_to_display("street_number", ATTR_STREET_NUMBER)
+ add_to_display("street", ATTR_STREET)
+
+ # Add remaining location details
+ for option_key, attr_key in {
+ "city": ATTR_CITY,
+ "county": ATTR_COUNTY,
+ "state": ATTR_REGION,
+ "region": ATTR_REGION,
+ "postal_code": ATTR_POSTAL_CODE,
+ "country": ATTR_COUNTRY,
+ "formatted_address": ATTR_FORMATTED_ADDRESS,
+ }.items():
+ add_to_display(option_key, attr_key)
+
+ # Handle "do_not_reorder" option
if "do_not_reorder" in display_options:
user_display = []
display_options.remove("do_not_reorder")
for option in display_options:
- if option == "state":
- target_option = "region"
- if option == "place_neighborhood":
- target_option = "place_neighbourhood"
- if option in locals():
- user_display.append(target_option)
+ attr_key = (
+ "region"
+ if option == "state"
+ else "place_neighbourhood"
+ if option == "place_neighborhood"
+ else option
+ )
+ if not self._is_attr_blank(attr_key):
+ user_display.append(self._get_attr_safe_str(attr_key))
+ # Set the final state
if user_display:
- self._set_attr(ATTR_NATIVE_VALUE, ", ".join(item for item in user_display))
+ self._set_attr(ATTR_NATIVE_VALUE, ", ".join(user_display))
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) New State from Display Options: "
- f"{self._get_attr(ATTR_NATIVE_VALUE)}"
+ "(%s) New State from Display Options: %s",
+ self._get_attr(CONF_NAME),
+ self._get_attr(ATTR_NATIVE_VALUE),
)
- async def _async_get_extended_attr(self):
- if not self._is_attr_blank(ATTR_OSM_ID) and not self._is_attr_blank(
- ATTR_OSM_TYPE
- ):
- if (self._get_attr(ATTR_OSM_TYPE)).lower() == "node":
+ async def _async_get_extended_attr(self) -> None:
+ if not self._is_attr_blank(ATTR_OSM_ID) and not self._is_attr_blank(ATTR_OSM_TYPE):
+ if self._get_attr_safe_str(ATTR_OSM_TYPE).lower() == "node":
osm_type_abbr = "N"
- elif (self._get_attr(ATTR_OSM_TYPE)).lower() == "way":
+ elif self._get_attr_safe_str(ATTR_OSM_TYPE).lower() == "way":
osm_type_abbr = "W"
- elif (self._get_attr(ATTR_OSM_TYPE)).lower() == "relation":
+ elif self._get_attr_safe_str(ATTR_OSM_TYPE).lower() == "relation":
osm_type_abbr = "R"
- osm_details_url = (
+ osm_details_url: str = (
"https://nominatim.openstreetmap.org/lookup?osm_ids="
f"{osm_type_abbr}{self._get_attr(ATTR_OSM_ID)}"
"&format=json&addressdetails=1&extratags=1&namedetails=1"
- f"&email={self._get_attr(CONF_API_KEY) if not self._is_attr_blank(
- CONF_API_KEY) else ''}"
- f"&accept-language={self._get_attr(CONF_LANGUAGE)
- if not self._is_attr_blank(CONF_LANGUAGE) else ''}"
+ f"&email={
+ self._get_attr(CONF_API_KEY) if not self._is_attr_blank(CONF_API_KEY) else ''
+ }"
+ f"&accept-language={
+ self._get_attr(CONF_LANGUAGE) if not self._is_attr_blank(CONF_LANGUAGE) else ''
+ }"
)
await self._hass.async_add_executor_job(
self._get_dict_from_url,
@@ -1930,27 +1954,27 @@ async def _async_get_extended_attr(self):
if (
not self._is_attr_blank(ATTR_OSM_DETAILS_DICT)
- and "extratags" in (self._get_attr(ATTR_OSM_DETAILS_DICT))
- and (self._get_attr(ATTR_OSM_DETAILS_DICT)).get("extratags")
- is not None
+ and "extratags" in self._get_attr_safe_dict(ATTR_OSM_DETAILS_DICT)
+ and self._get_attr_safe_dict(ATTR_OSM_DETAILS_DICT).get("extratags") is not None
and "wikidata"
- in (self._get_attr(ATTR_OSM_DETAILS_DICT)).get("extratags")
- and (self._get_attr(ATTR_OSM_DETAILS_DICT))
- .get("extratags")
+ in self._get_attr_safe_dict(ATTR_OSM_DETAILS_DICT).get("extratags", {})
+ and self._get_attr_safe_dict(ATTR_OSM_DETAILS_DICT)
+ .get("extratags", {})
.get("wikidata")
is not None
):
self._set_attr(
ATTR_WIKIDATA_ID,
- (self._get_attr(ATTR_OSM_DETAILS_DICT))
- .get("extratags")
+ self._get_attr_safe_dict(ATTR_OSM_DETAILS_DICT)
+ .get("extratags", {})
.get("wikidata"),
)
self._set_attr(ATTR_WIKIDATA_DICT, {})
if not self._is_attr_blank(ATTR_WIKIDATA_ID):
- wikidata_url = f"https://www.wikidata.org/wiki/Special:EntityData/{
- self._get_attr(ATTR_WIKIDATA_ID)}.json"
+ wikidata_url: str = f"https://www.wikidata.org/wiki/Special:EntityData/{
+ self._get_attr(ATTR_WIKIDATA_ID)
+ }.json"
await self._hass.async_add_executor_job(
self._get_dict_from_url,
wikidata_url,
@@ -1958,9 +1982,9 @@ async def _async_get_extended_attr(self):
ATTR_WIKIDATA_DICT,
)
- async def _async_fire_event_data(self, prev_last_place_name):
- _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) Building Event Data")
- event_data = {}
+ async def _async_fire_event_data(self, prev_last_place_name: str) -> None:
+ _LOGGER.debug("(%s) Building Event Data", self._get_attr(CONF_NAME))
+ event_data: MutableMapping[str, Any] = {}
if not self._is_attr_blank(CONF_NAME):
event_data.update({"entity": self._get_attr(CONF_NAME)})
if not self._is_attr_blank(ATTR_PREVIOUS_STATE):
@@ -1976,9 +2000,7 @@ async def _async_fire_event_data(self, prev_last_place_name):
not self._is_attr_blank(ATTR_LAST_PLACE_NAME)
and self._get_attr(ATTR_LAST_PLACE_NAME) != prev_last_place_name
):
- event_data.update(
- {ATTR_LAST_PLACE_NAME: self._get_attr(ATTR_LAST_PLACE_NAME)}
- )
+ event_data.update({ATTR_LAST_PLACE_NAME: self._get_attr(ATTR_LAST_PLACE_NAME)})
if self._get_attr(CONF_EXTENDED_ATTR):
for attr in EXTENDED_ATTRIBUTE_LIST:
@@ -1987,46 +2009,45 @@ async def _async_fire_event_data(self, prev_last_place_name):
self._hass.bus.fire(EVENT_TYPE, event_data)
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) Event Details [event_type: "
- f"{DOMAIN}_state_update]: {event_data}"
+ "(%s) Event Details [event_type: %s_state_update]: %s",
+ self._get_attr(CONF_NAME),
+ DOMAIN,
+ event_data,
)
_LOGGER.info(
- f"({self._get_attr(CONF_NAME)}) Event Fired [event_type: "
- f"{DOMAIN}_state_update]"
+ "(%s) Event Fired [event_type: %s_state_update]", self._get_attr(CONF_NAME), DOMAIN
)
- def _write_sensor_to_json(self, name, filename):
- sensor_attributes = copy.deepcopy(self._internal_attr)
- for k, v in list(sensor_attributes.items()):
+ def _write_sensor_to_json(self, name: str, filename: str) -> None:
+ sensor_attributes: MutableMapping[str, Any] = copy.deepcopy(self._internal_attr)
+ for k, v in sensor_attributes.items():
if isinstance(v, (datetime)):
# _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) Removing Sensor Attribute: {k}")
sensor_attributes.pop(k)
# _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) Sensor Attributes to Save: {sensor_attributes}")
try:
- with open(
- os.path.join(PLACES_JSON_FOLDER, filename),
- "w",
- ) as jsonfile:
+ json_file_path: Path = Path(self._json_folder) / filename
+ with json_file_path.open("w") as jsonfile:
json.dump(sensor_attributes, jsonfile)
except OSError as e:
_LOGGER.debug(
- f"({name}) OSError writing sensor to JSON "
- f"({filename}): {e.__class__.__qualname__}: {e}"
- )
- except Exception as e:
- _LOGGER.debug(
- f"({name}) Unknown Exception writing sensor to JSON "
- f"({filename}): {e.__class__.__qualname__}: {e}"
+ "(%s) OSError writing sensor to JSON (%s): %s: %s",
+ name,
+ filename,
+ e.__class__.__qualname__,
+ e,
)
- async def _async_get_initial_last_place_name(self):
+ async def _async_get_initial_last_place_name(self) -> None:
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) Previous State: "
- f"{self._get_attr(ATTR_PREVIOUS_STATE)}"
+ "(%s) Previous State: %s",
+ self._get_attr(CONF_NAME),
+ self._get_attr(ATTR_PREVIOUS_STATE),
)
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) Previous last_place_name: "
- f"{self._get_attr(ATTR_LAST_PLACE_NAME)}"
+ "(%s) Previous last_place_name: %s",
+ self._get_attr(CONF_NAME),
+ self._get_attr(ATTR_LAST_PLACE_NAME),
)
if not await self._async_in_zone():
@@ -2035,13 +2056,14 @@ async def _async_get_initial_last_place_name(self):
# If place name is set
self._set_attr(ATTR_LAST_PLACE_NAME, self._get_attr(ATTR_PLACE_NAME))
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) Previous place is Place Name, "
- f"last_place_name is set: {self._get_attr(ATTR_LAST_PLACE_NAME)}"
+ "(%s) Previous place is Place Name, last_place_name is set: %s",
+ self._get_attr(CONF_NAME),
+ self._get_attr(ATTR_LAST_PLACE_NAME),
)
else:
# If blank, keep previous last_place_name
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) Previous Place Name is None, keeping prior"
+ "(%s) Previous Place Name is None, keeping prior", self._get_attr(CONF_NAME)
)
else:
# Previously In a Zone
@@ -2050,22 +2072,22 @@ async def _async_get_initial_last_place_name(self):
self._get_attr(ATTR_DEVICETRACKER_ZONE_NAME),
)
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) Previous Place is Zone: "
- f"{self._get_attr(ATTR_LAST_PLACE_NAME)}"
+ "(%s) Previous Place is Zone: %s",
+ self._get_attr(CONF_NAME),
+ self._get_attr(ATTR_LAST_PLACE_NAME),
)
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) last_place_name (Initial): "
- f"{self._get_attr(ATTR_LAST_PLACE_NAME)}"
+ "(%s) last_place_name (Initial): %s",
+ self._get_attr(CONF_NAME),
+ self._get_attr(ATTR_LAST_PLACE_NAME),
)
- async def _async_update_coordinates_and_distance(self):
- last_distance_traveled_m = self._get_attr(ATTR_DISTANCE_FROM_HOME_M)
+ async def _async_update_coordinates_and_distance(self) -> int:
+ last_distance_traveled_m: float = self._get_attr_safe_float(ATTR_DISTANCE_FROM_HOME_M)
proceed_with_update = 1
# 0: False. 1: True. 2: False, but set direction of travel to stationary
- if not self._is_attr_blank(ATTR_LATITUDE) and not self._is_attr_blank(
- ATTR_LONGITUDE
- ):
+ if not self._is_attr_blank(ATTR_LATITUDE) and not self._is_attr_blank(ATTR_LONGITUDE):
self._set_attr(
ATTR_LOCATION_CURRENT,
f"{self._get_attr(ATTR_LATITUDE)},{self._get_attr(ATTR_LONGITUDE)}",
@@ -2075,16 +2097,14 @@ async def _async_update_coordinates_and_distance(self):
):
self._set_attr(
ATTR_LOCATION_PREVIOUS,
- f"{self._get_attr(ATTR_LATITUDE_OLD)},"
- f"{self._get_attr(ATTR_LONGITUDE_OLD)}",
+ f"{self._get_attr(ATTR_LATITUDE_OLD)},{self._get_attr(ATTR_LONGITUDE_OLD)}",
)
if not self._is_attr_blank(ATTR_HOME_LATITUDE) and not self._is_attr_blank(
ATTR_HOME_LONGITUDE
):
self._set_attr(
ATTR_HOME_LOCATION,
- f"{self._get_attr(ATTR_HOME_LATITUDE)},"
- f"{self._get_attr(ATTR_HOME_LONGITUDE)}",
+ f"{self._get_attr(ATTR_HOME_LATITUDE)},{self._get_attr(ATTR_HOME_LONGITUDE)}",
)
if (
@@ -2096,20 +2116,20 @@ async def _async_update_coordinates_and_distance(self):
self._set_attr(
ATTR_DISTANCE_FROM_HOME_M,
distance(
- float(self._get_attr(ATTR_LATITUDE)),
- float(self._get_attr(ATTR_LONGITUDE)),
- float(self._get_attr(ATTR_HOME_LATITUDE)),
- float(self._get_attr(ATTR_HOME_LONGITUDE)),
+ float(self._get_attr_safe_str(ATTR_LATITUDE)),
+ float(self._get_attr_safe_str(ATTR_LONGITUDE)),
+ float(self._get_attr_safe_str(ATTR_HOME_LATITUDE)),
+ float(self._get_attr_safe_str(ATTR_HOME_LONGITUDE)),
),
)
if not self._is_attr_blank(ATTR_DISTANCE_FROM_HOME_M):
self._set_attr(
ATTR_DISTANCE_FROM_HOME_KM,
- round(self._get_attr(ATTR_DISTANCE_FROM_HOME_M) / 1000, 3),
+ round(self._get_attr_safe_float(ATTR_DISTANCE_FROM_HOME_M) / 1000, 3),
)
self._set_attr(
ATTR_DISTANCE_FROM_HOME_MI,
- round(self._get_attr(ATTR_DISTANCE_FROM_HOME_M) / 1609, 3),
+ round(self._get_attr_safe_float(ATTR_DISTANCE_FROM_HOME_M) / 1609, 3),
)
if not self._is_attr_blank(ATTR_LATITUDE_OLD) and not self._is_attr_blank(
@@ -2118,27 +2138,24 @@ async def _async_update_coordinates_and_distance(self):
self._set_attr(
ATTR_DISTANCE_TRAVELED_M,
distance(
- float(self._get_attr(ATTR_LATITUDE)),
- float(self._get_attr(ATTR_LONGITUDE)),
- float(self._get_attr(ATTR_LATITUDE_OLD)),
- float(self._get_attr(ATTR_LONGITUDE_OLD)),
+ float(self._get_attr_safe_str(ATTR_LATITUDE)),
+ float(self._get_attr_safe_str(ATTR_LONGITUDE)),
+ float(self._get_attr_safe_str(ATTR_LATITUDE_OLD)),
+ float(self._get_attr_safe_str(ATTR_LONGITUDE_OLD)),
),
)
if not self._is_attr_blank(ATTR_DISTANCE_TRAVELED_M):
self._set_attr(
ATTR_DISTANCE_TRAVELED_MI,
round(
- self._get_attr(ATTR_DISTANCE_TRAVELED_M) / 1609,
+ self._get_attr_safe_float(ATTR_DISTANCE_TRAVELED_M) / 1609,
3,
),
)
- # if self._get_attr(ATTR_DISTANCE_TRAVELED_M) <= 100: # in meters
- # self._set_attr(ATTR_DIRECTION_OF_TRAVEL, "stationary")
- # elif last_distance_traveled_m > self._get_attr(ATTR_DISTANCE_FROM_HOME_M):
- if last_distance_traveled_m > self._get_attr(ATTR_DISTANCE_FROM_HOME_M):
+ if last_distance_traveled_m > self._get_attr_safe_float(ATTR_DISTANCE_FROM_HOME_M):
self._set_attr(ATTR_DIRECTION_OF_TRAVEL, "towards home")
- elif last_distance_traveled_m < self._get_attr(
+ elif last_distance_traveled_m < self._get_attr_safe_float(
ATTR_DISTANCE_FROM_HOME_M
):
self._set_attr(ATTR_DIRECTION_OF_TRAVEL, "away from home")
@@ -2150,439 +2167,403 @@ async def _async_update_coordinates_and_distance(self):
self._set_attr(ATTR_DISTANCE_TRAVELED_MI, 0)
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) Previous Location: "
- f"{self._get_attr(ATTR_LOCATION_PREVIOUS)}"
+ "(%s) Previous Location: %s",
+ self._get_attr(CONF_NAME),
+ self._get_attr(ATTR_LOCATION_PREVIOUS),
)
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) Current Location: "
- f"{self._get_attr(ATTR_LOCATION_CURRENT)}"
+ "(%s) Current Location: %s",
+ self._get_attr(CONF_NAME),
+ self._get_attr(ATTR_LOCATION_CURRENT),
)
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) Home Location: "
- f"{self._get_attr(ATTR_HOME_LOCATION)}"
+ "(%s) Home Location: %s",
+ self._get_attr(CONF_NAME),
+ self._get_attr(ATTR_HOME_LOCATION),
)
_LOGGER.info(
- f"({self._get_attr(CONF_NAME)}) Distance from home "
- f"[{(self._get_attr(CONF_HOME_ZONE)).split('.')[1]}]: "
- f"{self._get_attr(ATTR_DISTANCE_FROM_HOME_KM)} km"
+ "(%s) Distance from home [%s]: %s km",
+ self._get_attr(CONF_NAME),
+ self._get_attr_safe_str(CONF_HOME_ZONE).split(".")[1],
+ self._get_attr(ATTR_DISTANCE_FROM_HOME_KM),
)
_LOGGER.info(
- f"({self._get_attr(CONF_NAME)}) Travel Direction: "
- f"{self._get_attr(ATTR_DIRECTION_OF_TRAVEL)}"
+ "(%s) Travel Direction: %s",
+ self._get_attr(CONF_NAME),
+ self._get_attr(ATTR_DIRECTION_OF_TRAVEL),
)
_LOGGER.info(
- f"({self._get_attr(CONF_NAME)}) Meters traveled since last update: "
- f"{round(self._get_attr(ATTR_DISTANCE_TRAVELED_M), 1)}"
+ "(%s) Meters traveled since last update: %s",
+ self._get_attr(CONF_NAME),
+ round(self._get_attr_safe_float(ATTR_DISTANCE_TRAVELED_M), 1),
)
else:
proceed_with_update = 0
# 0: False. 1: True. 2: False, but set direction of travel to stationary
_LOGGER.info(
- f"({self._get_attr(CONF_NAME)}) "
- "Problem with updated lat/long, not performing update: "
- f"old_latitude={self._get_attr(ATTR_LATITUDE_OLD)}, "
- f"old_longitude={self._get_attr(ATTR_LONGITUDE_OLD)}, "
- f"new_latitude={self._get_attr(ATTR_LATITUDE)}, "
- f"new_longitude={self._get_attr(ATTR_LONGITUDE)}, "
- f"home_latitude={self._get_attr(ATTR_HOME_LATITUDE)}, "
- f"home_longitude={self._get_attr(ATTR_HOME_LONGITUDE)}"
+ "(%s) Problem with updated lat/long, not performing update: "
+ "old_latitude=%s, old_longitude=%s, "
+ "new_latitude=%s, new_longitude=%s, "
+ "home_latitude=%s, home_longitude=%s",
+ self._get_attr(CONF_NAME),
+ self._get_attr(ATTR_LATITUDE_OLD),
+ self._get_attr(ATTR_LONGITUDE_OLD),
+ self._get_attr(ATTR_LATITUDE),
+ self._get_attr(ATTR_LONGITUDE),
+ self._get_attr(ATTR_HOME_LATITUDE),
+ self._get_attr(ATTR_HOME_LONGITUDE),
)
return proceed_with_update
- async def _async_finalize_last_place_name(self, prev_last_place_name=None):
+ async def _async_finalize_last_place_name(self, prev_last_place_name: str) -> None:
if self._get_attr(ATTR_INITIAL_UPDATE):
self._set_attr(ATTR_LAST_PLACE_NAME, prev_last_place_name)
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) "
- "Runnining initial update after load, using prior last_place_name"
+ "(%s) Runnining initial update after load, using prior last_place_name",
+ self._get_attr(CONF_NAME),
)
elif self._get_attr(ATTR_LAST_PLACE_NAME) == self._get_attr(
ATTR_PLACE_NAME
- ) or self._get_attr(ATTR_LAST_PLACE_NAME) == self._get_attr(
- ATTR_DEVICETRACKER_ZONE_NAME
- ):
+ ) or self._get_attr(ATTR_LAST_PLACE_NAME) == self._get_attr(ATTR_DEVICETRACKER_ZONE_NAME):
# If current place name/zone are the same as previous, keep older last_place_name
self._set_attr(ATTR_LAST_PLACE_NAME, prev_last_place_name)
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) "
- "Initial last_place_name is same as new: place_name="
- f"{self._get_attr(ATTR_PLACE_NAME)} or devicetracker_zone_name="
- f"{self._get_attr(ATTR_DEVICETRACKER_ZONE_NAME)}, "
- "keeping previous last_place_name"
+ "(%s) Initial last_place_name is same as new: place_name=%s or devicetracker_zone_name=%s, "
+ "keeping previous last_place_name",
+ self._get_attr(CONF_NAME),
+ self._get_attr(ATTR_PLACE_NAME),
+ self._get_attr(ATTR_DEVICETRACKER_ZONE_NAME),
)
else:
- _LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) Keeping initial last_place_name"
- )
+ _LOGGER.debug("(%s) Keeping initial last_place_name", self._get_attr(CONF_NAME))
_LOGGER.info(
- f"({self._get_attr(CONF_NAME)}) last_place_name: "
- f"{self._get_attr(ATTR_LAST_PLACE_NAME)}"
+ "(%s) last_place_name: %s",
+ self._get_attr(CONF_NAME),
+ self._get_attr(ATTR_LAST_PLACE_NAME),
)
- async def _async_do_update(self, reason):
+ async def _async_do_update(self, reason: str) -> None:
"""Get the latest data and updates the states."""
-
_LOGGER.info(
- f"({self._get_attr(CONF_NAME)}) Starting {reason} Update (Tracked Entity: "
- f"{self._get_attr(CONF_DEVICETRACKER_ID)})"
+ "(%s) Starting %s Update (Tracked Entity: %s)",
+ self._get_attr(CONF_NAME),
+ reason,
+ self._get_attr(CONF_DEVICETRACKER_ID),
)
- if self._hass.config.time_zone is not None:
- now = datetime.now(tz=ZoneInfo(str(self._hass.config.time_zone)))
+ now: datetime = await self._get_current_time()
+ previous_attr: MutableMapping[str, Any] = copy.deepcopy(self._internal_attr)
+
+ await self._update_entity_name_and_cleanup()
+ await self._update_previous_state()
+ await self._update_old_coordinates()
+ prev_last_place_name = self._get_attr_safe_str(ATTR_LAST_PLACE_NAME)
+
+ # 0: False. 1: True. 2: False, but set direction of travel to stationary
+ proceed_with_update: int = await self._check_device_tracker_and_update_coords()
+
+ if proceed_with_update == 1:
+ proceed_with_update = await self._determine_update_criteria()
+
+ if proceed_with_update == 1:
+ await self._process_osm_update(now=now)
+
+ if await self._should_update_state(now=now):
+ await self._handle_state_update(now=now, prev_last_place_name=prev_last_place_name)
+ else:
+ _LOGGER.info(
+ "(%s) No entity update needed, Previous State = New State",
+ self._get_attr(CONF_NAME),
+ )
+ await self._rollback_update(previous_attr, now, proceed_with_update)
else:
- now = datetime.now()
- previous_attr = copy.deepcopy(self._internal_attr)
+ await self._rollback_update(previous_attr, now, proceed_with_update)
- await self._async_check_for_updated_entity_name()
+ self._set_attr(ATTR_LAST_UPDATED, now.isoformat(sep=" ", timespec="seconds"))
+ _LOGGER.info("(%s) End of Update", self._get_attr(CONF_NAME))
+
+ async def _should_update_state(self, now: datetime) -> bool:
+ prev_state: str = self._get_attr_safe_str(ATTR_PREVIOUS_STATE)
+ native_value: str = self._get_attr_safe_str(ATTR_NATIVE_VALUE)
+ tracker_zone: str = self._get_attr_safe_str(ATTR_DEVICETRACKER_ZONE)
+
+ if (
+ (
+ not self._is_attr_blank(ATTR_PREVIOUS_STATE)
+ and not self._is_attr_blank(ATTR_NATIVE_VALUE)
+ and prev_state.lower().strip() != native_value.lower().strip()
+ and prev_state.replace(" ", "").lower().strip() != native_value.lower().strip()
+ and prev_state.lower().strip() != tracker_zone.lower().strip()
+ )
+ or self._is_attr_blank(ATTR_PREVIOUS_STATE)
+ or self._is_attr_blank(ATTR_NATIVE_VALUE)
+ or self._get_attr(ATTR_INITIAL_UPDATE)
+ ):
+ return True
+ return False
+
+ async def _handle_state_update(self, now: datetime, prev_last_place_name: str) -> None:
+ if self._get_attr(CONF_EXTENDED_ATTR):
+ await self._async_get_extended_attr()
+ self._set_attr(ATTR_SHOW_DATE, False)
await self._async_cleanup_attributes()
- # _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) Previous entity attributes: {self._internal_attr}")
- if not self._is_attr_blank(ATTR_NATIVE_VALUE) and self._get_attr(
- CONF_SHOW_TIME
+
+ if not self._is_attr_blank(ATTR_NATIVE_VALUE):
+ current_time: str = f"{now.hour:02}:{now.minute:02}"
+ if self._get_attr(CONF_SHOW_TIME):
+ state: str = await Places._async_clear_since_from_state(
+ self._get_attr_safe_str(ATTR_NATIVE_VALUE)
+ )
+ self._set_attr(ATTR_NATIVE_VALUE, f"{state[: 255 - 14]} (since {current_time})")
+ else:
+ self._set_attr(ATTR_NATIVE_VALUE, self._get_attr_safe_str(ATTR_NATIVE_VALUE)[:255])
+ _LOGGER.info(
+ "(%s) New State: %s",
+ self._get_attr(CONF_NAME),
+ self._get_attr(ATTR_NATIVE_VALUE),
+ )
+ else:
+ self._clear_attr(ATTR_NATIVE_VALUE)
+ _LOGGER.warning("(%s) New State is None", self._get_attr(CONF_NAME))
+
+ if not self._is_attr_blank(ATTR_NATIVE_VALUE):
+ self._attr_native_value = self._get_attr(ATTR_NATIVE_VALUE)
+ else:
+ self._attr_native_value = None
+
+ await self._async_fire_event_data(prev_last_place_name=prev_last_place_name)
+ self._set_attr(ATTR_INITIAL_UPDATE, False)
+ await self._hass.async_add_executor_job(
+ self._write_sensor_to_json,
+ self._get_attr(CONF_NAME),
+ self._get_attr(ATTR_JSON_FILENAME),
+ )
+
+ async def _rollback_update(
+ self, previous_attr: MutableMapping[str, Any], now: datetime, proceed_with_update: int
+ ) -> None:
+ self._internal_attr = previous_attr
+ _LOGGER.debug(
+ "(%s) Reverting attributes back to before the update started", self._get_attr(CONF_NAME)
+ )
+ changed_diff_sec = await self._async_get_seconds_from_last_change(now=now)
+ if (
+ proceed_with_update == 2
+ and self._get_attr(ATTR_DIRECTION_OF_TRAVEL) != "stationary"
+ and changed_diff_sec >= 60
+ ):
+ await self._async_change_dot_to_stationary(now=now, changed_diff_sec=changed_diff_sec)
+ if (
+ self._get_attr(CONF_SHOW_TIME)
+ and changed_diff_sec >= 86399
+ and not self._get_attr(ATTR_SHOW_DATE)
):
+ await self._async_change_show_time_to_date()
+
+ async def _get_current_time(self) -> datetime:
+ if self._hass.config.time_zone:
+ return datetime.now(tz=ZoneInfo(str(self._hass.config.time_zone)))
+ return datetime.now()
+
+ async def _update_entity_name_and_cleanup(self) -> None:
+ await self._async_check_for_updated_entity_name()
+ await self._async_cleanup_attributes()
+
+ async def _update_previous_state(self) -> None:
+ if not self._is_attr_blank(ATTR_NATIVE_VALUE) and self._get_attr(CONF_SHOW_TIME):
self._set_attr(
ATTR_PREVIOUS_STATE,
- await self._async_clear_since_from_state(
- str(self._get_attr(ATTR_NATIVE_VALUE))
+ await Places._async_clear_since_from_state(
+ orig_state=self._get_attr_safe_str(ATTR_NATIVE_VALUE)
),
)
else:
self._set_attr(ATTR_PREVIOUS_STATE, self._get_attr(ATTR_NATIVE_VALUE))
- if self._is_float(self._get_attr(ATTR_LATITUDE)):
+
+ async def _update_old_coordinates(self) -> None:
+ if _is_float(self._get_attr(ATTR_LATITUDE)):
self._set_attr(ATTR_LATITUDE_OLD, str(self._get_attr(ATTR_LATITUDE)))
- if self._is_float(self._get_attr(ATTR_LONGITUDE)):
+ if _is_float(self._get_attr(ATTR_LONGITUDE)):
self._set_attr(ATTR_LONGITUDE_OLD, str(self._get_attr(ATTR_LONGITUDE)))
- prev_last_place_name = self._get_attr(ATTR_LAST_PLACE_NAME)
- proceed_with_update = await self._async_is_devicetracker_set()
- # 0: False. 1: True. 2: False, but set direction of travel to stationary
+ async def _check_device_tracker_and_update_coords(self) -> int:
+ proceed_with_update: int = await self._async_is_devicetracker_set()
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) "
- f"[is_devicetracker_set] proceed_with_update: {proceed_with_update}"
+ "(%s) [is_devicetracker_set] proceed_with_update: %s",
+ self._get_attr(CONF_NAME),
+ proceed_with_update,
)
if proceed_with_update == 1:
- # 0: False. 1: True. 2: False, but set direction of travel to stationary
- if self._is_float(
- self._hass.states.get(
- self._get_attr(CONF_DEVICETRACKER_ID)
- ).attributes.get(CONF_LATITUDE)
- ):
- self._set_attr(
- ATTR_LATITUDE,
- str(
- self._hass.states.get(
- self._get_attr(CONF_DEVICETRACKER_ID)
- ).attributes.get(CONF_LATITUDE)
- ),
- )
- if self._is_float(
- self._hass.states.get(
- self._get_attr(CONF_DEVICETRACKER_ID)
- ).attributes.get(CONF_LONGITUDE)
- ):
- self._set_attr(
- ATTR_LONGITUDE,
- str(
- self._hass.states.get(
- self._get_attr(CONF_DEVICETRACKER_ID)
- ).attributes.get(CONF_LONGITUDE)
- ),
- )
+ await self._update_coordinates()
proceed_with_update = await self._async_get_gps_accuracy()
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) "
- f"[is_devicetracker_set] proceed_with_update: {proceed_with_update}"
+ "(%s) [is_devicetracker_set] proceed_with_update: %s",
+ self._get_attr(CONF_NAME),
+ proceed_with_update,
)
+ return proceed_with_update
+ async def _update_coordinates(self) -> None:
+ device_tracker = self._hass.states.get(self._get_attr(CONF_DEVICETRACKER_ID))
+ if _is_float(device_tracker.attributes.get(CONF_LATITUDE)):
+ self._set_attr(ATTR_LATITUDE, str(device_tracker.attributes.get(CONF_LATITUDE)))
+ if _is_float(device_tracker.attributes.get(CONF_LONGITUDE)):
+ self._set_attr(ATTR_LONGITUDE, str(device_tracker.attributes.get(CONF_LONGITUDE)))
+
+ async def _determine_update_criteria(self) -> int:
+ await self._async_get_initial_last_place_name()
+ await self._async_get_zone_details()
+ proceed_with_update = await self._async_update_coordinates_and_distance()
+ _LOGGER.debug(
+ "(%s) [update_coordinates_and_distance] proceed_with_update: %s",
+ self._get_attr(CONF_NAME),
+ proceed_with_update,
+ )
if proceed_with_update == 1:
- # 0: False. 1: True. 2: False, but set direction of travel to stationary
- await self._async_get_initial_last_place_name()
- await self._async_get_zone_details()
- proceed_with_update = await self._async_update_coordinates_and_distance()
+ proceed_with_update = await self._async_determine_if_update_needed()
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) "
- "[update_coordinates_and_distance] proceed_with_update: "
- f"{proceed_with_update}"
+ "(%s) [determine_if_update_needed] proceed_with_update: %s",
+ self._get_attr(CONF_NAME),
+ proceed_with_update,
)
+ return proceed_with_update
- if proceed_with_update == 1:
- # 0: False. 1: True. 2: False, but set direction of travel to stationary
- proceed_with_update = await self._async_determine_if_update_needed()
+ async def _process_osm_update(self, now: datetime) -> None:
+ _LOGGER.info(
+ "(%s) Meets criteria, proceeding with OpenStreetMap query",
+ self._get_attr(CONF_NAME),
+ )
+ _LOGGER.info(
+ "(%s) Tracked Entity Zone: %s",
+ self._get_attr(CONF_NAME),
+ self._get_attr(ATTR_DEVICETRACKER_ZONE),
+ )
+
+ await self._async_reset_attributes()
+ await self._async_get_map_link()
+ await self._query_osm_and_finalize(now=now)
+
+ async def _query_osm_and_finalize(self, now: datetime) -> None:
+ osm_url: str = await self._build_osm_url()
+ await self._hass.async_add_executor_job(
+ self._get_dict_from_url, osm_url, "OpenStreetMaps", ATTR_OSM_DICT
+ )
+ if not self._is_attr_blank(ATTR_OSM_DICT):
+ await self._async_parse_osm_dict()
+ await self._async_finalize_last_place_name(
+ self._get_attr_safe_str(ATTR_LAST_PLACE_NAME)
+ )
+ await self._process_display_options()
+ self._set_attr(ATTR_LAST_CHANGED, now.isoformat(sep=" ", timespec="seconds"))
+
+ async def _process_display_options(self) -> None:
+ display_options: list[str] = []
+ if not self._is_attr_blank(ATTR_DISPLAY_OPTIONS):
+ options_array = self._get_attr_safe_str(ATTR_DISPLAY_OPTIONS).split(",")
+ for option in options_array:
+ display_options.extend([option.strip()])
+ self._set_attr(ATTR_DISPLAY_OPTIONS_LIST, display_options)
+
+ await self._async_get_driving_status()
+
+ if "formatted_place" in display_options:
+ await self._async_build_formatted_place()
+ self._set_attr(
+ ATTR_NATIVE_VALUE,
+ self._get_attr(ATTR_FORMATTED_PLACE),
+ )
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) "
- f"[determine_if_update_needed] proceed_with_update: {
- proceed_with_update}"
+ "(%s) New State using formatted_place: %s",
+ self._get_attr(CONF_NAME),
+ self._get_attr(ATTR_NATIVE_VALUE),
)
- if proceed_with_update == 1:
- # 0: False. 1: True. 2: False, but set direction of travel to stationary
- _LOGGER.info(
- f"({self._get_attr(CONF_NAME)}) Meets criteria, proceeding with OpenStreetMap query"
+ elif any(
+ ext in (self._get_attr_safe_str(ATTR_DISPLAY_OPTIONS)) for ext in ["(", ")", "[", "]"]
+ ):
+ self._clear_attr(ATTR_DISPLAY_OPTIONS_LIST)
+ display_options = []
+ self._adv_options_state_list = []
+ self._street_num_i = -1
+ self._street_i = -1
+ self._temp_i = 0
+ _LOGGER.debug(
+ "(%s) Initial Advanced Display Options: %s",
+ self._get_attr(CONF_NAME),
+ self._get_attr(ATTR_DISPLAY_OPTIONS),
)
- _LOGGER.info(
- f"({self._get_attr(CONF_NAME)}) Tracked Entity Zone: "
- f"{self._get_attr(ATTR_DEVICETRACKER_ZONE)}"
- # f" / Skipped Updates: {self._get_attr(ATTR_UPDATES_SKIPPED)}"
+ await self._async_build_from_advanced_options(
+ self._get_attr_safe_str(ATTR_DISPLAY_OPTIONS)
)
-
- await self._async_reset_attributes()
- await self._async_get_map_link()
-
- osm_url = (
- "https://nominatim.openstreetmap.org/reverse?format=jsonv2&lat="
- f"{self._get_attr(ATTR_LATITUDE)}"
- f"&lon={self._get_attr(ATTR_LONGITUDE)}"
- f"&accept-language={self._get_attr(CONF_LANGUAGE)
- if not self._is_attr_blank(CONF_LANGUAGE) else ''}"
- "&addressdetails=1&namedetails=1&zoom=18&limit=1"
- f"&email={self._get_attr(CONF_API_KEY) if not self._is_attr_blank(
- CONF_API_KEY) else ''}"
+ _LOGGER.debug(
+ "(%s) Back from initial advanced build: %s",
+ self._get_attr(CONF_NAME),
+ self._adv_options_state_list,
)
-
- await self._hass.async_add_executor_job(
- self._get_dict_from_url, osm_url, "OpenStreetMaps", ATTR_OSM_DICT
+ await self._async_compile_state_from_advanced_options()
+ elif not await self._async_in_zone():
+ await self._async_build_state_from_display_options()
+ elif (
+ "zone" in display_options and not self._is_attr_blank(ATTR_DEVICETRACKER_ZONE)
+ ) or self._is_attr_blank(ATTR_DEVICETRACKER_ZONE_NAME):
+ self._set_attr(
+ ATTR_NATIVE_VALUE,
+ self._get_attr(ATTR_DEVICETRACKER_ZONE),
)
-
- if not self._is_attr_blank(ATTR_OSM_DICT):
- await self._async_parse_osm_dict()
- await self._async_finalize_last_place_name(prev_last_place_name)
-
- display_options = []
- if not self._is_attr_blank(ATTR_DISPLAY_OPTIONS):
- options_array = (self._get_attr(ATTR_DISPLAY_OPTIONS)).split(",")
- for option in options_array:
- display_options.append(option.strip())
- self._set_attr(ATTR_DISPLAY_OPTIONS_LIST, display_options)
-
- await self._async_get_driving_status()
-
- if "formatted_place" in display_options:
- await self._async_build_formatted_place()
- self._set_attr(
- ATTR_NATIVE_VALUE,
- self._get_attr(ATTR_FORMATTED_PLACE),
- )
- _LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) New State using formatted_place: "
- f"{self._get_attr(ATTR_NATIVE_VALUE)}"
- )
-
- elif any(
- ext in (self._get_attr(ATTR_DISPLAY_OPTIONS))
- for ext in ["(", ")", "[", "]"]
- ):
- # Replace place option with expanded definition
- # temp_opt = self._get_attr(ATTR_DISPLAY_OPTIONS)
- # re.sub(
- # r"place(?=[\[\(\]\)\,\s])",
- # "place_name,place_category(-,place),place_type(-,yes),neighborhood,street_number,street",
- # temp_opt,
- # )
- # self._set_attr(ATTR_DISPLAY_OPTIONS, temp_opt)
- self._clear_attr(ATTR_DISPLAY_OPTIONS_LIST)
- display_options = None
- self._adv_options_state_list = []
- self._street_num_i = -1
- self._street_i = -1
- self._temp_i = 0
- _LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) Initial Advanced Display Options: "
- f"{self._get_attr(ATTR_DISPLAY_OPTIONS)}"
- )
-
- await self._async_build_from_advanced_options(
- self._get_attr(ATTR_DISPLAY_OPTIONS)
- )
- # _LOGGER.debug(
- # f"({self._get_attr(CONF_NAME)}) Back from initial advanced build: "
- # + f"{self._adv_options_state_list}"
- # )
- await self._async_compile_state_from_advanced_options()
- elif not await self._async_in_zone():
- await self._async_build_state_from_display_options()
- elif (
- "zone" in display_options
- and not self._is_attr_blank(ATTR_DEVICETRACKER_ZONE)
- ) or self._is_attr_blank(ATTR_DEVICETRACKER_ZONE_NAME):
- self._set_attr(
- ATTR_NATIVE_VALUE,
- self._get_attr(ATTR_DEVICETRACKER_ZONE),
- )
- _LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) New State from Tracked Entity Zone: "
- f"{self._get_attr(ATTR_NATIVE_VALUE)}"
- )
- elif not self._is_attr_blank(ATTR_DEVICETRACKER_ZONE_NAME):
- self._set_attr(
- ATTR_NATIVE_VALUE,
- self._get_attr(ATTR_DEVICETRACKER_ZONE_NAME),
- )
- _LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) New State from Tracked Entity Zone Name: "
- f"{self._get_attr(ATTR_NATIVE_VALUE)}"
- )
- current_time = f"{now.hour:02}:{now.minute:02}"
- self._set_attr(
- ATTR_LAST_CHANGED, str(now.isoformat(sep=" ", timespec="seconds"))
- )
-
- # Final check to see if the New State is different from the Previous State and should update or not.
- # If not, attributes are reset to what they were before the update started.
-
- if (
- (
- not self._is_attr_blank(ATTR_PREVIOUS_STATE)
- and not self._is_attr_blank(ATTR_NATIVE_VALUE)
- and (self._get_attr(ATTR_PREVIOUS_STATE)).lower().strip()
- != (self._get_attr(ATTR_NATIVE_VALUE)).lower().strip()
- and (self._get_attr(ATTR_PREVIOUS_STATE))
- .replace(" ", "")
- .lower()
- .strip()
- != (self._get_attr(ATTR_NATIVE_VALUE)).lower().strip()
- and self._get_attr(ATTR_PREVIOUS_STATE).lower().strip()
- != (self._get_attr(ATTR_DEVICETRACKER_ZONE)).lower().strip()
- )
- or self._is_attr_blank(ATTR_PREVIOUS_STATE)
- or self._is_attr_blank(ATTR_NATIVE_VALUE)
- or self._get_attr(ATTR_INITIAL_UPDATE)
- ):
- if self._get_attr(CONF_EXTENDED_ATTR):
- await self._async_get_extended_attr()
- self._set_attr(ATTR_SHOW_DATE, False)
- await self._async_cleanup_attributes()
- if not self._is_attr_blank(ATTR_NATIVE_VALUE):
- if self._get_attr(CONF_SHOW_TIME):
- self._set_attr(
- ATTR_NATIVE_VALUE,
- str(
- await self._async_clear_since_from_state(
- str(self._get_attr(ATTR_NATIVE_VALUE))
- )
- )[: 255 - 14]
- + " (since "
- + current_time
- + ")",
- )
- else:
- self._set_attr(
- ATTR_NATIVE_VALUE,
- self._get_attr(ATTR_NATIVE_VALUE)[:255],
- )
- _LOGGER.info(
- f"({self._get_attr(CONF_NAME)}) New State: "
- f"{self._get_attr(ATTR_NATIVE_VALUE)}"
- )
- else:
- self._clear_attr(ATTR_NATIVE_VALUE)
- _LOGGER.warning(
- f"({self._get_attr(CONF_NAME)}) New State is None"
- )
- if not self._is_attr_blank(ATTR_NATIVE_VALUE):
- self._attr_native_value = self._get_attr(ATTR_NATIVE_VALUE)
- else:
- self._attr_native_value = None
- await self._async_fire_event_data(prev_last_place_name)
- self._set_attr(ATTR_INITIAL_UPDATE, False)
- await self._hass.async_add_executor_job(
- self._write_sensor_to_json,
- self._get_attr(CONF_NAME),
- self._get_attr(ATTR_JSON_FILENAME),
- )
- else:
- self._internal_attr = previous_attr
- _LOGGER.info(
- f"({self._get_attr(CONF_NAME)}) "
- "No entity update needed, Previous State = New State"
- )
- _LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) "
- "Reverting attributes back to before the update started"
- )
-
- changed_diff_sec = await self._async_get_seconds_from_last_change(
- now
- )
- if (
- self._get_attr(ATTR_DIRECTION_OF_TRAVEL) != "stationary"
- and changed_diff_sec >= 60
- ):
- await self._async_change_dot_to_stationary(
- now, changed_diff_sec
- )
- if (
- self._get_attr(CONF_SHOW_TIME)
- and changed_diff_sec >= 86399
- and self._get_attr(ATTR_SHOW_DATE) is False
- ):
- await self._async_change_show_time_to_date()
- else:
- self._internal_attr = previous_attr
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) "
- "Reverting attributes back to before the update started"
+ "(%s) New State from Tracked Entity Zone: %s",
+ self._get_attr(CONF_NAME),
+ self._get_attr(ATTR_NATIVE_VALUE),
+ )
+ elif not self._is_attr_blank(ATTR_DEVICETRACKER_ZONE_NAME):
+ self._set_attr(
+ ATTR_NATIVE_VALUE,
+ self._get_attr(ATTR_DEVICETRACKER_ZONE_NAME),
+ )
+ _LOGGER.debug(
+ "(%s) New State from Tracked Entity Zone Name: %s",
+ self._get_attr(CONF_NAME),
+ self._get_attr(ATTR_NATIVE_VALUE),
)
- changed_diff_sec = await self._async_get_seconds_from_last_change(now)
- if (
- proceed_with_update == 2
- and self._get_attr(ATTR_DIRECTION_OF_TRAVEL) != "stationary"
- and changed_diff_sec >= 60
- ):
- # 0: False. 1: True. 2: False, but set direction of travel to stationary
- await self._async_change_dot_to_stationary(now, changed_diff_sec)
- if (
- self._get_attr(CONF_SHOW_TIME)
- and changed_diff_sec >= 86399
- and self._get_attr(ATTR_SHOW_DATE) is False
- ):
- await self._async_change_show_time_to_date()
-
- self._set_attr(
- ATTR_LAST_UPDATED, str(now.isoformat(sep=" ", timespec="seconds"))
- )
- # _LOGGER.debug(f"({self._get_attr(CONF_NAME)}) Final entity attributes: {await self._async__internal_attr}")
- _LOGGER.info(f"({self._get_attr(CONF_NAME)}) End of Update")
+ async def _build_osm_url(self) -> str:
+ """Build the OpenStreetMap query URL."""
+ base_url = "https://nominatim.openstreetmap.org/reverse?format=json"
+ lat: str = self._get_attr_safe_str(ATTR_LATITUDE)
+ lon: str = self._get_attr_safe_str(ATTR_LONGITUDE)
+ lang: str = self._get_attr_safe_str(CONF_LANGUAGE)
+ email: str = self._get_attr_safe_str(CONF_API_KEY)
+ return f"{base_url}&lat={lat}&lon={lon}&accept-language={lang}&addressdetails=1&namedetails=1&zoom=18&limit=1&email={email}"
- async def _async_change_dot_to_stationary(self, now, changed_diff_sec):
+ async def _async_change_dot_to_stationary(self, now: datetime, changed_diff_sec: int) -> None:
self._set_attr(ATTR_DIRECTION_OF_TRAVEL, "stationary")
- self._set_attr(
- ATTR_LAST_CHANGED, str(now.isoformat(sep=" ", timespec="seconds"))
- )
+ self._set_attr(ATTR_LAST_CHANGED, now.isoformat(sep=" ", timespec="seconds"))
await self._hass.async_add_executor_job(
self._write_sensor_to_json,
self._get_attr(CONF_NAME),
self._get_attr(ATTR_JSON_FILENAME),
)
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) "
- "Updating direction of travel to stationary (Last changed "
- f"{int(changed_diff_sec)} seconds ago)"
+ "(%s) Updating direction of travel to stationary (Last changed %s seconds ago)",
+ self._get_attr(CONF_NAME),
+ int(changed_diff_sec),
)
- async def _async_change_show_time_to_date(self):
- if not self._is_attr_blank(ATTR_NATIVE_VALUE) and self._get_attr(
- CONF_SHOW_TIME
- ):
- # localedate = str(locale.nl_langinfo(locale.D_FMT)).replace(" ", "")
- # if localedate.lower().endswith("%y"):
- # localemmdd = localedate[:-3]
- # elif localedate.lower().startswith("%y"):
- # localemmdd = localedate[3:]
- # else:
+ async def _async_change_show_time_to_date(self) -> None:
+ if not self._is_attr_blank(ATTR_NATIVE_VALUE) and self._get_attr(CONF_SHOW_TIME):
if self._get_attr(CONF_DATE_FORMAT) == "dd/mm":
dateformat = "%d/%m"
else:
dateformat = "%m/%d"
- mmddstring = (
- datetime.fromisoformat(self._get_attr(ATTR_LAST_CHANGED))
+ mmddstring: str = (
+ datetime.fromisoformat(self._get_attr_safe_str(ATTR_LAST_CHANGED))
.strftime(f"{dateformat}")
.replace(" ", "")[:5]
)
self._set_attr(
ATTR_NATIVE_VALUE,
- f"{await self._async_clear_since_from_state(str(self._get_attr(ATTR_NATIVE_VALUE)))}"
- + f" (since {mmddstring})",
+ f"{await Places._async_clear_since_from_state(self._get_attr_safe_str(ATTR_NATIVE_VALUE))} (since {mmddstring})",
)
if not self._is_attr_blank(ATTR_NATIVE_VALUE):
@@ -2596,24 +2577,25 @@ async def _async_change_show_time_to_date(self):
self._get_attr(ATTR_JSON_FILENAME),
)
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) "
- "Updating state to show date instead of time since last change"
+ "(%s) Updating state to show date instead of time since last change",
+ self._get_attr(CONF_NAME),
)
_LOGGER.debug(
- f"({self._get_attr(CONF_NAME)}) New State: "
- f"{self._get_attr(ATTR_NATIVE_VALUE)}"
+ "(%s) New State: %s", self._get_attr(CONF_NAME), self._get_attr(ATTR_NATIVE_VALUE)
)
- async def _async_get_seconds_from_last_change(self, now):
+ async def _async_get_seconds_from_last_change(self, now: datetime) -> int:
if self._is_attr_blank(ATTR_LAST_CHANGED):
return 3600
try:
- last_changed = datetime.fromisoformat(self._get_attr(ATTR_LAST_CHANGED))
+ last_changed: datetime = datetime.fromisoformat(
+ self._get_attr_safe_str(ATTR_LAST_CHANGED)
+ )
except (TypeError, ValueError) as e:
_LOGGER.warning(
- f"Error converting Last Changed date/time "
- f"({self._get_attr(ATTR_LAST_CHANGED)}) "
- f"into datetime: {repr(e)}"
+ "Error converting Last Changed date/time (%s) into datetime: %r",
+ self._get_attr(ATTR_LAST_CHANGED),
+ e,
)
return 3600
else:
@@ -2624,25 +2606,22 @@ async def _async_get_seconds_from_last_change(self, now):
changed_diff_sec = (datetime.now() - last_changed).total_seconds()
except (TypeError, OverflowError) as e:
_LOGGER.warning(
- "Error calculating the seconds between last change to now: "
- f"{repr(e)}"
+ "Error calculating the seconds between last change to now: %r", e
)
return 3600
except OverflowError as e:
- _LOGGER.warning(
- "Error calculating the seconds between last change to now: "
- f"{repr(e)}"
- )
+ _LOGGER.warning("Error calculating the seconds between last change to now: %r", e)
return 3600
- return changed_diff_sec
+ return int(changed_diff_sec)
- async def _async_reset_attributes(self):
- """Resets attributes."""
+ async def _async_reset_attributes(self) -> None:
+ """Reset sensor attributes."""
for attr in RESET_ATTRIBUTE_LIST:
self._clear_attr(attr)
- # self._set_attr(ATTR_UPDATES_SKIPPED, 0)
await self._async_cleanup_attributes()
class PlacesNoRecorder(Places):
+ """Places Class without the HA Recorder."""
+
_unrecorded_attributes = frozenset({MATCH_ALL})
diff --git a/icon/map-search-outline.svg b/icon/map-search-outline.svg
index 436219ff..86d1df93 100644
--- a/icon/map-search-outline.svg
+++ b/icon/map-search-outline.svg
@@ -1 +1 @@
-
\ No newline at end of file
+
diff --git a/pyproject.toml b/pyproject.toml
new file mode 100644
index 00000000..7b803eab
--- /dev/null
+++ b/pyproject.toml
@@ -0,0 +1,289 @@
+[tool.codespell]
+skip = "*.py,*.toml,CHANGELOG.*"
+quiet-level = 3
+ignore-words-list = "hass"
+
+[tool.coverage.report]
+show_missing = true
+fail_under = 80
+
+[tool.mypy]
+python_version = "3.13"
+follow_imports = "silent"
+ignore_missing_imports = true
+platform = "linux"
+show_error_codes = true
+local_partial_types = true
+strict_equality = true
+no_implicit_optional = true
+warn_incomplete_stub = true
+warn_redundant_casts = true
+warn_unused_configs = true
+warn_unused_ignores = true
+enable_error_code = ["deprecated", "ignore-without-code", "redundant-self", "truthy-iterable"]
+disable_error_code = ["annotation-unchecked", "import-not-found", "import-untyped", "call-arg"]
+extra_checks = false
+check_untyped_defs = true
+disallow_incomplete_defs = false
+disallow_subclassing_any = false
+disallow_untyped_calls = false
+disallow_untyped_decorators = false
+disallow_untyped_defs = false
+warn_return_any = false
+warn_unreachable = false
+
+[tool.pytest.ini_options]
+testpaths = [
+ "tests",
+]
+norecursedirs = [
+ ".git",
+ "testing_config",
+]
+log_format = "%(asctime)s.%(msecs)03d %(levelname)-8s %(threadName)s %(name)s:%(filename)s:%(lineno)s %(message)s"
+log_date_format = "%Y-%m-%d %H:%M:%S"
+# log_cli = true # Enable live logging to the console
+# log_level = "DEBUG" # Set the logging level to DEBUG
+asyncio_mode = "auto"
+asyncio_default_fixture_loop_scope = "function"
+timeout = 30
+addopts = "-rA"
+
+[tool.ruff]
+line-length = 100
+indent-width = 4
+fix = true
+force-exclude = true
+target-version = "py313"
+required-version = ">=0.8.0"
+
+[tool.ruff.format]
+quote-style = "double"
+indent-style = "space"
+skip-magic-trailing-comma = false
+line-ending = "lf"
+docstring-code-format = true
+docstring-code-line-length = "dynamic"
+
+[tool.ruff.lint]
+select = [
+ "A001", # Variable {name} is shadowing a Python builtin
+ "ASYNC210", # Async functions should not call blocking HTTP methods
+ "ASYNC220", # Async functions should not create subprocesses with blocking methods
+ "ASYNC221", # Async functions should not run processes with blocking methods
+ "ASYNC222", # Async functions should not wait on processes with blocking methods
+ "ASYNC230", # Async functions should not open files with blocking methods like open
+ "ASYNC251", # Async functions should not call time.sleep
+ "B002", # Python does not support the unary prefix increment
+ "B005", # Using .strip() with multi-character strings is misleading
+ "B007", # Loop control variable {name} not used within loop body
+ "B014", # Exception handler with duplicate exception
+ "B015", # Pointless comparison. Did you mean to assign a value? Otherwise, prepend assert or remove it.
+ "B017", # pytest.raises(BaseException) should be considered evil
+ "B018", # Found useless attribute access. Either assign it to a variable or remove it.
+ "B023", # Function definition does not bind loop variable {name}
+ "B026", # Star-arg unpacking after a keyword argument is strongly discouraged
+ "B032", # Possible unintentional type annotation (using :). Did you mean to assign (using =)?
+ "B904", # Use raise from to specify exception cause
+ "B905", # zip() without an explicit strict= parameter
+ "BLE",
+ "C", # complexity
+ "COM818", # Trailing comma on bare tuple prohibited
+ "D", # docstrings
+ "DTZ003", # Use datetime.now(tz=) instead of datetime.utcnow()
+ "DTZ004", # Use datetime.fromtimestamp(ts, tz=) instead of datetime.utcfromtimestamp(ts)
+ "E", # pycodestyle
+ "F", # pyflakes/autoflake
+ "F541", # f-string without any placeholders
+ "FLY", # flynt
+ "FURB", # refurb
+ "G", # flake8-logging-format
+ "I", # isort
+ "INP", # flake8-no-pep420
+ "ISC", # flake8-implicit-str-concat
+ "ICN001", # import concentions; {name} should be imported as {asname}
+ "LOG", # flake8-logging
+ "N804", # First argument of a class method should be named cls
+ "N805", # First argument of a method should be named self
+ "N815", # Variable {name} in class scope should not be mixedCase
+ "PERF", # Perflint
+ "PGH", # pygrep-hooks
+ "PIE", # flake8-pie
+ "PL", # pylint
+ "PT", # flake8-pytest-style
+ "PTH", # flake8-pathlib
+ "PYI", # flake8-pyi
+ "RET", # flake8-return
+ "RSE", # flake8-raise
+ "RUF005", # Consider iterable unpacking instead of concatenation
+ "RUF006", # Store a reference to the return value of asyncio.create_task
+ "RUF010", # Use explicit conversion flag
+ "RUF013", # PEP 484 prohibits implicit Optional
+ "RUF017", # Avoid quadratic list summation
+ "RUF018", # Avoid assignment expressions in assert statements
+ "RUF019", # Unnecessary key check before dictionary access
+ # "RUF100", # Unused `noqa` directive; temporarily every now and then to clean them up
+ "S102", # Use of exec detected
+ "S103", # bad-file-permissions
+ "S108", # hardcoded-temp-file
+ "S306", # suspicious-mktemp-usage
+ "S307", # suspicious-eval-usage
+ "S313", # suspicious-xmlc-element-tree-usage
+ "S314", # suspicious-xml-element-tree-usage
+ "S315", # suspicious-xml-expat-reader-usage
+ "S316", # suspicious-xml-expat-builder-usage
+ "S317", # suspicious-xml-sax-usage
+ "S318", # suspicious-xml-mini-dom-usage
+ "S319", # suspicious-xml-pull-dom-usage
+ "S320", # suspicious-xmle-tree-usage
+ "S601", # paramiko-call
+ "S602", # subprocess-popen-with-shell-equals-true
+ "S604", # call-with-shell-equals-true
+ "S608", # hardcoded-sql-expression
+ "S609", # unix-command-wildcard-injection
+ "SIM", # flake8-simplify
+ "SLF", # flake8-self
+ "SLOT", # flake8-slots
+ "T100", # Trace found: {name} used
+ "T20", # flake8-print
+ "TC", # flake8-type-checking
+ "TID", # Tidy imports
+ "TRY", # tryceratops
+ "UP", # pyupgrade
+ "UP031", # Use format specifiers instead of percent format
+ "UP032", # Use f-string instead of `format` call
+ "W", # pycodestyle
+]
+
+ignore = [
+ # "C901", # McCabe cyclomatic complexity
+ "D202", # No blank lines allowed after function docstring
+ "D203", # 1 blank line required before class docstring
+ "D213", # Multi-line docstring summary should start at the second line
+ "D406", # Section name should end with a newline
+ "D407", # Section name underlining
+ "E501", # line too long
+
+ "PLC1901", # {existing} can be simplified to {replacement} as an empty string is falsey; too many false positives
+ "PLR0904", # Too many public methods
+ "PLR0911", # Too many return statements ({returns} > {max_returns})
+ "PLR0912", # Too many branches ({branches} > {max_branches})
+ "PLR0913", # Too many arguments to function call ({c_args} > {max_args})
+ "PLR0914", # Too many local variables
+ "PLR0915", # Too many statements ({statements} > {max_statements})
+ "PLR0916", # Too many Boolean expressions
+ "PLR0917", # Too many positional arguments
+ "PLR1702", # Too many nested blocks
+ "PLR2004", # Magic value used in comparison, consider replacing {value} with a constant variable
+ "PLW2901", # Outer {outer_kind} variable {name} overwritten by inner {inner_kind} target
+ "PT011", # pytest.raises({exception}) is too broad, set the `match` parameter or use a more specific exception
+ "PT018", # Assertion should be broken down into multiple parts
+ "RUF001", # String contains ambiguous unicode character.
+ "RUF002", # Docstring contains ambiguous unicode character.
+ "RUF003", # Comment contains ambiguous unicode character.
+ "RUF015", # Prefer next(...) over single element slice
+ "SIM102", # Use a single if statement instead of nested if statements
+ "SIM103", # Return the condition {condition} directly
+ "SIM108", # Use ternary operator {contents} instead of if-else-block
+ "SIM115", # Use context handler for opening files
+
+ # Moving imports into type-checking blocks can mess with pytest.patch()
+ "TC001", # Move application import {} into a type-checking block
+ "TC002", # Move third-party import {} into a type-checking block
+ "TC003", # Move standard library import {} into a type-checking block
+
+ "TRY003", # Avoid specifying long messages outside the exception class
+ "TRY400", # Use `logging.exception` instead of `logging.error`
+
+ # May conflict with the formatter, https://docs.astral.sh/ruff/formatter/#conflicting-lint-rules
+ "W191",
+ "E111",
+ "E114",
+ "E117",
+ "D206",
+ "D300",
+ "Q",
+ "COM812",
+ "COM819",
+ "ISC001",
+
+ # Disabled because ruff does not understand type of __all__ generated by a function
+ "PLE0605"
+]
+fixable = ["ALL"]
+
+[tool.ruff.lint.flake8-import-conventions.extend-aliases]
+voluptuous = "vol"
+"homeassistant.components.air_quality.PLATFORM_SCHEMA" = "AIR_QUALITY_PLATFORM_SCHEMA"
+"homeassistant.components.alarm_control_panel.PLATFORM_SCHEMA" = "ALARM_CONTROL_PANEL_PLATFORM_SCHEMA"
+"homeassistant.components.binary_sensor.PLATFORM_SCHEMA" = "BINARY_SENSOR_PLATFORM_SCHEMA"
+"homeassistant.components.button.PLATFORM_SCHEMA" = "BUTTON_PLATFORM_SCHEMA"
+"homeassistant.components.calendar.PLATFORM_SCHEMA" = "CALENDAR_PLATFORM_SCHEMA"
+"homeassistant.components.camera.PLATFORM_SCHEMA" = "CAMERA_PLATFORM_SCHEMA"
+"homeassistant.components.climate.PLATFORM_SCHEMA" = "CLIMATE_PLATFORM_SCHEMA"
+"homeassistant.components.conversation.PLATFORM_SCHEMA" = "CONVERSATION_PLATFORM_SCHEMA"
+"homeassistant.components.cover.PLATFORM_SCHEMA" = "COVER_PLATFORM_SCHEMA"
+"homeassistant.components.date.PLATFORM_SCHEMA" = "DATE_PLATFORM_SCHEMA"
+"homeassistant.components.datetime.PLATFORM_SCHEMA" = "DATETIME_PLATFORM_SCHEMA"
+"homeassistant.components.device_tracker.PLATFORM_SCHEMA" = "DEVICE_TRACKER_PLATFORM_SCHEMA"
+"homeassistant.components.event.PLATFORM_SCHEMA" = "EVENT_PLATFORM_SCHEMA"
+"homeassistant.components.fan.PLATFORM_SCHEMA" = "FAN_PLATFORM_SCHEMA"
+"homeassistant.components.geo_location.PLATFORM_SCHEMA" = "GEO_LOCATION_PLATFORM_SCHEMA"
+"homeassistant.components.humidifier.PLATFORM_SCHEMA" = "HUMIDIFIER_PLATFORM_SCHEMA"
+"homeassistant.components.image.PLATFORM_SCHEMA" = "IMAGE_PLATFORM_SCHEMA"
+"homeassistant.components.image_processing.PLATFORM_SCHEMA" = "IMAGE_PROCESSING_PLATFORM_SCHEMA"
+"homeassistant.components.lawn_mower.PLATFORM_SCHEMA" = "LAWN_MOWER_PLATFORM_SCHEMA"
+"homeassistant.components.light.PLATFORM_SCHEMA" = "LIGHT_PLATFORM_SCHEMA"
+"homeassistant.components.lock.PLATFORM_SCHEMA" = "LOCK_PLATFORM_SCHEMA"
+"homeassistant.components.media_player.PLATFORM_SCHEMA" = "MEDIA_PLAYER_PLATFORM_SCHEMA"
+"homeassistant.components.notify.PLATFORM_SCHEMA" = "NOTIFY_PLATFORM_SCHEMA"
+"homeassistant.components.number.PLATFORM_SCHEMA" = "NUMBER_PLATFORM_SCHEMA"
+"homeassistant.components.remote.PLATFORM_SCHEMA" = "REMOTE_PLATFORM_SCHEMA"
+"homeassistant.components.scene.PLATFORM_SCHEMA" = "SCENE_PLATFORM_SCHEMA"
+"homeassistant.components.select.PLATFORM_SCHEMA" = "SELECT_PLATFORM_SCHEMA"
+"homeassistant.components.sensor.PLATFORM_SCHEMA" = "SENSOR_PLATFORM_SCHEMA"
+"homeassistant.components.siren.PLATFORM_SCHEMA" = "SIREN_PLATFORM_SCHEMA"
+"homeassistant.components.stt.PLATFORM_SCHEMA" = "STT_PLATFORM_SCHEMA"
+"homeassistant.components.switch.PLATFORM_SCHEMA" = "SWITCH_PLATFORM_SCHEMA"
+"homeassistant.components.text.PLATFORM_SCHEMA" = "TEXT_PLATFORM_SCHEMA"
+"homeassistant.components.time.PLATFORM_SCHEMA" = "TIME_PLATFORM_SCHEMA"
+"homeassistant.components.todo.PLATFORM_SCHEMA" = "TODO_PLATFORM_SCHEMA"
+"homeassistant.components.tts.PLATFORM_SCHEMA" = "TTS_PLATFORM_SCHEMA"
+"homeassistant.components.vacuum.PLATFORM_SCHEMA" = "VACUUM_PLATFORM_SCHEMA"
+"homeassistant.components.valve.PLATFORM_SCHEMA" = "VALVE_PLATFORM_SCHEMA"
+"homeassistant.components.update.PLATFORM_SCHEMA" = "UPDATE_PLATFORM_SCHEMA"
+"homeassistant.components.wake_word.PLATFORM_SCHEMA" = "WAKE_WORD_PLATFORM_SCHEMA"
+"homeassistant.components.water_heater.PLATFORM_SCHEMA" = "WATER_HEATER_PLATFORM_SCHEMA"
+"homeassistant.components.weather.PLATFORM_SCHEMA" = "WEATHER_PLATFORM_SCHEMA"
+"homeassistant.core.DOMAIN" = "HOMEASSISTANT_DOMAIN"
+"homeassistant.helpers.area_registry" = "ar"
+"homeassistant.helpers.category_registry" = "cr"
+"homeassistant.helpers.config_validation" = "cv"
+"homeassistant.helpers.device_registry" = "dr"
+"homeassistant.helpers.entity_registry" = "er"
+"homeassistant.helpers.floor_registry" = "fr"
+"homeassistant.helpers.issue_registry" = "ir"
+"homeassistant.helpers.label_registry" = "lr"
+"homeassistant.util.dt" = "dt_util"
+
+[tool.ruff.lint.flake8-pytest-style]
+fixture-parentheses = false
+mark-parentheses = false
+
+[tool.ruff.lint.flake8-tidy-imports.banned-api]
+"async_timeout".msg = "use asyncio.timeout instead"
+"pytz".msg = "use zoneinfo instead"
+
+[tool.ruff.lint.isort]
+force-sort-within-sections = true
+known-first-party = [
+ "homeassistant",
+]
+combine-as-imports = true
+split-on-trailing-comma = false
+
+[tool.ruff.lint.mccabe]
+max-complexity = 25
+
+[tool.ruff.lint.pydocstyle]
+property-decorators = ["propcache.cached_property"]
diff --git a/requirements.txt b/requirements.txt
new file mode 100644
index 00000000..5d723c84
--- /dev/null
+++ b/requirements.txt
@@ -0,0 +1,8 @@
+aiohttp_cors
+mypy
+ruff
+types-python-dateutil
+types-PyMySQL
+types-PyYAML
+types-pyRFC3339
+types-requests