2020-06-15 10:02:25 +00:00
|
|
|
"""The Met Office integration."""
|
2022-09-28 14:37:35 +00:00
|
|
|
from __future__ import annotations
|
2020-06-15 10:02:25 +00:00
|
|
|
|
2021-06-27 19:04:42 +00:00
|
|
|
import asyncio
|
2020-06-15 10:02:25 +00:00
|
|
|
import logging
|
2022-09-28 14:37:35 +00:00
|
|
|
import re
|
|
|
|
from typing import Any
|
2020-06-15 10:02:25 +00:00
|
|
|
|
2021-06-27 19:04:42 +00:00
|
|
|
import datapoint
|
|
|
|
|
2020-06-15 10:02:25 +00:00
|
|
|
from homeassistant.config_entries import ConfigEntry
|
2021-12-06 03:06:35 +00:00
|
|
|
from homeassistant.const import (
|
|
|
|
CONF_API_KEY,
|
|
|
|
CONF_LATITUDE,
|
|
|
|
CONF_LONGITUDE,
|
|
|
|
CONF_NAME,
|
|
|
|
Platform,
|
|
|
|
)
|
2022-09-28 14:37:35 +00:00
|
|
|
from homeassistant.core import HomeAssistant, callback
|
2020-06-15 10:02:25 +00:00
|
|
|
from homeassistant.exceptions import ConfigEntryNotReady
|
2023-03-01 07:02:51 +00:00
|
|
|
from homeassistant.helpers import device_registry as dr, entity_registry as er
|
2021-11-23 21:45:23 +00:00
|
|
|
from homeassistant.helpers.entity import DeviceInfo
|
2020-06-15 10:02:25 +00:00
|
|
|
from homeassistant.helpers.update_coordinator import DataUpdateCoordinator
|
|
|
|
|
|
|
|
from .const import (
|
|
|
|
DEFAULT_SCAN_INTERVAL,
|
|
|
|
DOMAIN,
|
2021-06-27 19:04:42 +00:00
|
|
|
METOFFICE_COORDINATES,
|
|
|
|
METOFFICE_DAILY_COORDINATOR,
|
|
|
|
METOFFICE_HOURLY_COORDINATOR,
|
2020-06-15 10:02:25 +00:00
|
|
|
METOFFICE_NAME,
|
2021-06-27 19:04:42 +00:00
|
|
|
MODE_3HOURLY,
|
|
|
|
MODE_DAILY,
|
2020-06-15 10:02:25 +00:00
|
|
|
)
|
|
|
|
from .data import MetOfficeData
|
2021-06-27 19:04:42 +00:00
|
|
|
from .helpers import fetch_data, fetch_site
|
2020-06-15 10:02:25 +00:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2021-12-06 03:06:35 +00:00
|
|
|
PLATFORMS = [Platform.SENSOR, Platform.WEATHER]
|
2020-06-15 10:02:25 +00:00
|
|
|
|
|
|
|
|
2021-05-27 15:39:06 +00:00
|
|
|
async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
2020-06-15 10:02:25 +00:00
|
|
|
"""Set up a Met Office entry."""
|
|
|
|
|
|
|
|
latitude = entry.data[CONF_LATITUDE]
|
|
|
|
longitude = entry.data[CONF_LONGITUDE]
|
|
|
|
api_key = entry.data[CONF_API_KEY]
|
|
|
|
site_name = entry.data[CONF_NAME]
|
|
|
|
|
2022-09-28 14:37:35 +00:00
|
|
|
coordinates = f"{latitude}_{longitude}"
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def update_unique_id(
|
2023-03-01 07:02:51 +00:00
|
|
|
entity_entry: er.RegistryEntry,
|
2022-09-28 14:37:35 +00:00
|
|
|
) -> dict[str, Any] | None:
|
|
|
|
"""Update unique ID of entity entry."""
|
|
|
|
|
|
|
|
if entity_entry.domain != Platform.SENSOR:
|
|
|
|
return None
|
|
|
|
|
|
|
|
name_to_key = {
|
|
|
|
"Station Name": "name",
|
|
|
|
"Weather": "weather",
|
|
|
|
"Temperature": "temperature",
|
|
|
|
"Feels Like Temperature": "feels_like_temperature",
|
|
|
|
"Wind Speed": "wind_speed",
|
|
|
|
"Wind Direction": "wind_direction",
|
|
|
|
"Wind Gust": "wind_gust",
|
|
|
|
"Visibility": "visibility",
|
|
|
|
"Visibility Distance": "visibility_distance",
|
|
|
|
"UV Index": "uv",
|
|
|
|
"Probability of Precipitation": "precipitation",
|
|
|
|
"Humidity": "humidity",
|
|
|
|
}
|
|
|
|
|
|
|
|
match = re.search(f"(?P<name>.*)_{coordinates}.*", entity_entry.unique_id)
|
|
|
|
|
|
|
|
if match is None:
|
|
|
|
return None
|
|
|
|
|
|
|
|
if (name := match.group("name")) in name_to_key:
|
|
|
|
return {
|
|
|
|
"new_unique_id": entity_entry.unique_id.replace(name, name_to_key[name])
|
|
|
|
}
|
|
|
|
return None
|
|
|
|
|
2023-03-01 07:02:51 +00:00
|
|
|
await er.async_migrate_entries(hass, entry.entry_id, update_unique_id)
|
2022-09-28 14:37:35 +00:00
|
|
|
|
2021-06-27 19:04:42 +00:00
|
|
|
connection = datapoint.connection(api_key=api_key)
|
|
|
|
|
|
|
|
site = await hass.async_add_executor_job(
|
|
|
|
fetch_site, connection, latitude, longitude
|
|
|
|
)
|
|
|
|
if site is None:
|
2020-06-15 10:02:25 +00:00
|
|
|
raise ConfigEntryNotReady()
|
|
|
|
|
2021-06-27 19:04:42 +00:00
|
|
|
async def async_update_3hourly() -> MetOfficeData:
|
|
|
|
return await hass.async_add_executor_job(
|
|
|
|
fetch_data, connection, site, MODE_3HOURLY
|
|
|
|
)
|
|
|
|
|
|
|
|
async def async_update_daily() -> MetOfficeData:
|
|
|
|
return await hass.async_add_executor_job(
|
|
|
|
fetch_data, connection, site, MODE_DAILY
|
|
|
|
)
|
|
|
|
|
|
|
|
metoffice_hourly_coordinator = DataUpdateCoordinator(
|
|
|
|
hass,
|
|
|
|
_LOGGER,
|
|
|
|
name=f"MetOffice Hourly Coordinator for {site_name}",
|
|
|
|
update_method=async_update_3hourly,
|
|
|
|
update_interval=DEFAULT_SCAN_INTERVAL,
|
|
|
|
)
|
|
|
|
|
|
|
|
metoffice_daily_coordinator = DataUpdateCoordinator(
|
2020-06-15 10:02:25 +00:00
|
|
|
hass,
|
|
|
|
_LOGGER,
|
2021-06-27 19:04:42 +00:00
|
|
|
name=f"MetOffice Daily Coordinator for {site_name}",
|
|
|
|
update_method=async_update_daily,
|
2020-06-15 10:02:25 +00:00
|
|
|
update_interval=DEFAULT_SCAN_INTERVAL,
|
|
|
|
)
|
|
|
|
|
|
|
|
metoffice_hass_data = hass.data.setdefault(DOMAIN, {})
|
|
|
|
metoffice_hass_data[entry.entry_id] = {
|
2021-06-27 19:04:42 +00:00
|
|
|
METOFFICE_HOURLY_COORDINATOR: metoffice_hourly_coordinator,
|
|
|
|
METOFFICE_DAILY_COORDINATOR: metoffice_daily_coordinator,
|
2020-06-15 10:02:25 +00:00
|
|
|
METOFFICE_NAME: site_name,
|
2022-09-28 14:37:35 +00:00
|
|
|
METOFFICE_COORDINATES: coordinates,
|
2020-06-15 10:02:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
# Fetch initial data so we have data when entities subscribe
|
2021-06-27 19:04:42 +00:00
|
|
|
await asyncio.gather(
|
|
|
|
metoffice_hourly_coordinator.async_config_entry_first_refresh(),
|
|
|
|
metoffice_daily_coordinator.async_config_entry_first_refresh(),
|
|
|
|
)
|
2020-06-15 10:02:25 +00:00
|
|
|
|
2022-07-09 15:27:42 +00:00
|
|
|
await hass.config_entries.async_forward_entry_setups(entry, PLATFORMS)
|
2020-06-15 10:02:25 +00:00
|
|
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
2021-10-06 08:48:11 +00:00
|
|
|
async def async_unload_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
2020-06-15 10:02:25 +00:00
|
|
|
"""Unload a config entry."""
|
2021-04-27 16:49:13 +00:00
|
|
|
unload_ok = await hass.config_entries.async_unload_platforms(entry, PLATFORMS)
|
2020-06-15 10:02:25 +00:00
|
|
|
if unload_ok:
|
|
|
|
hass.data[DOMAIN].pop(entry.entry_id)
|
|
|
|
if not hass.data[DOMAIN]:
|
|
|
|
hass.data.pop(DOMAIN)
|
|
|
|
return unload_ok
|
2021-11-23 21:45:23 +00:00
|
|
|
|
|
|
|
|
|
|
|
def get_device_info(coordinates: str, name: str) -> DeviceInfo:
|
|
|
|
"""Return device registry information."""
|
|
|
|
return DeviceInfo(
|
2023-03-01 07:02:51 +00:00
|
|
|
entry_type=dr.DeviceEntryType.SERVICE,
|
2021-11-23 21:45:23 +00:00
|
|
|
identifiers={(DOMAIN, coordinates)},
|
|
|
|
manufacturer="Met Office",
|
|
|
|
name=f"Met Office {name}",
|
|
|
|
)
|