Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Update metoffice to use DataHub API #131425

Open
wants to merge 16 commits into
base: dev
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
69 changes: 11 additions & 58 deletions homeassistant/components/metoffice/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -4,10 +4,10 @@

import asyncio
import logging
import re
from typing import Any

import datapoint
import datapoint.Forecast
import datapoint.Manager

from homeassistant.config_entries import ConfigEntry
from homeassistant.const import (
Expand All @@ -17,9 +17,8 @@
CONF_NAME,
Platform,
)
from homeassistant.core import HomeAssistant, callback
from homeassistant.exceptions import ConfigEntryNotReady
from homeassistant.helpers import device_registry as dr, entity_registry as er
from homeassistant.core import HomeAssistant
from homeassistant.helpers import device_registry as dr
from homeassistant.helpers.device_registry import DeviceInfo
from homeassistant.helpers.update_coordinator import TimestampDataUpdateCoordinator

Expand All @@ -30,11 +29,8 @@
METOFFICE_DAILY_COORDINATOR,
METOFFICE_HOURLY_COORDINATOR,
METOFFICE_NAME,
MODE_3HOURLY,
MODE_DAILY,
)
from .data import MetOfficeData
from .helpers import fetch_data, fetch_site
from .helpers import fetch_data

_LOGGER = logging.getLogger(__name__)

Expand All @@ -51,67 +47,24 @@ async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:

coordinates = f"{latitude}_{longitude}"

@callback
def update_unique_id(
entity_entry: er.RegistryEntry,
) -> 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

await er.async_migrate_entries(hass, entry.entry_id, update_unique_id)

connection = datapoint.connection(api_key=api_key)

site = await hass.async_add_executor_job(
fetch_site, connection, latitude, longitude
)
if site is None:
raise ConfigEntryNotReady
connection = datapoint.Manager.Manager(api_key=api_key)

async def async_update_3hourly() -> MetOfficeData:
async def async_update_hourly() -> datapoint.Forecast:
return await hass.async_add_executor_job(
fetch_data, connection, site, MODE_3HOURLY
fetch_data, connection, latitude, longitude, "hourly"
)

async def async_update_daily() -> MetOfficeData:
async def async_update_daily() -> datapoint.Forecast:
return await hass.async_add_executor_job(
fetch_data, connection, site, MODE_DAILY
fetch_data, connection, latitude, longitude, "daily"
)

metoffice_hourly_coordinator = TimestampDataUpdateCoordinator(
hass,
_LOGGER,
config_entry=entry,
name=f"MetOffice Hourly Coordinator for {site_name}",
update_method=async_update_3hourly,
update_method=async_update_hourly,
update_interval=DEFAULT_SCAN_INTERVAL,
)

Expand Down
109 changes: 85 additions & 24 deletions homeassistant/components/metoffice/config_flow.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,10 +2,14 @@

from __future__ import annotations

from collections.abc import Mapping
import logging
from typing import Any

import datapoint
from datapoint.exceptions import APIException
import datapoint.Manager
from requests import HTTPError
import voluptuous as vol

from homeassistant.config_entries import ConfigFlow, ConfigFlowResult
Expand All @@ -15,30 +19,41 @@
from homeassistant.helpers import config_validation as cv

from .const import DOMAIN
from .helpers import fetch_site

_LOGGER = logging.getLogger(__name__)


async def validate_input(hass: HomeAssistant, data: dict[str, Any]) -> dict[str, str]:
async def validate_input(
hass: HomeAssistant, latitude: float, longitude: float, api_key: str
) -> dict[str, Any]:
"""Validate that the user input allows us to connect to DataPoint.

Data has the keys from DATA_SCHEMA with values provided by the user.
"""
latitude = data[CONF_LATITUDE]
longitude = data[CONF_LONGITUDE]
api_key = data[CONF_API_KEY]

connection = datapoint.connection(api_key=api_key)

site = await hass.async_add_executor_job(
fetch_site, connection, latitude, longitude
)
errors = {}
connection = datapoint.Manager.Manager(api_key=api_key)

try:
forecast = await hass.async_add_executor_job(
connection.get_forecast,
latitude,
longitude,
"daily",
False,
)

if site is None:
raise CannotConnect
except (HTTPError, APIException) as err:
if isinstance(err, HTTPError) and err.response.status_code == 401:
errors["base"] = "invalid_auth"
else:
errors["base"] = "cannot_connect"
except Exception:
_LOGGER.exception("Unexpected exception")
errors["base"] = "unknown"
else:
return {"site_name": forecast.name, "errors": errors}

return {"site_name": site.name}
return {"errors": errors}
avee87 marked this conversation as resolved.
Show resolved Hide resolved


class MetOfficeConfigFlow(ConfigFlow, domain=DOMAIN):
Expand All @@ -57,15 +72,17 @@ async def async_step_user(
)
self._abort_if_unique_id_configured()

try:
info = await validate_input(self.hass, user_input)
except CannotConnect:
errors["base"] = "cannot_connect"
except Exception:
_LOGGER.exception("Unexpected exception")
errors["base"] = "unknown"
else:
user_input[CONF_NAME] = info["site_name"]
result = await validate_input(
self.hass,
latitude=user_input[CONF_LATITUDE],
longitude=user_input[CONF_LONGITUDE],
api_key=user_input[CONF_API_KEY],
)

errors = result["errors"]
avee87 marked this conversation as resolved.
Show resolved Hide resolved

if not errors:
user_input[CONF_NAME] = result["site_name"]
return self.async_create_entry(
title=user_input[CONF_NAME], data=user_input
)
Expand All @@ -83,7 +100,51 @@ async def async_step_user(
)

return self.async_show_form(
step_id="user", data_schema=data_schema, errors=errors
step_id="user",
data_schema=data_schema,
errors=errors,
)

async def async_step_reauth(
self, entry_data: Mapping[str, Any]
) -> ConfigFlowResult:
"""Perform reauth upon an API authentication error."""
return await self.async_step_reauth_confirm()

async def async_step_reauth_confirm(
self, user_input: dict[str, Any] | None = None
) -> ConfigFlowResult:
"""Dialog that informs the user that reauth is required."""
errors = {}

entry = self._get_reauth_entry()
if user_input is not None:
result = await validate_input(
self.hass,
latitude=entry.data[CONF_LATITUDE],
longitude=entry.data[CONF_LONGITUDE],
api_key=user_input[CONF_API_KEY],
)

errors = result["errors"]

if not errors:
return self.async_update_reload_and_abort(
self._get_reauth_entry(),
data_updates=user_input,
)

return self.async_show_form(
step_id="reauth_confirm",
data_schema=vol.Schema(
{
vol.Required(CONF_API_KEY): str,
}
),
description_placeholders={
"docs_url": ("https://www.home-assistant.io/integrations/metoffice")
},
errors=errors,
)


Expand Down
76 changes: 46 additions & 30 deletions homeassistant/components/metoffice/const.py
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,17 @@
ATTR_CONDITION_SUNNY,
ATTR_CONDITION_WINDY,
ATTR_CONDITION_WINDY_VARIANT,
ATTR_FORECAST_CONDITION,
ATTR_FORECAST_NATIVE_APPARENT_TEMP,
ATTR_FORECAST_NATIVE_PRESSURE,
ATTR_FORECAST_NATIVE_TEMP,
ATTR_FORECAST_NATIVE_TEMP_LOW,
ATTR_FORECAST_NATIVE_WIND_GUST_SPEED,
ATTR_FORECAST_NATIVE_WIND_SPEED,
ATTR_FORECAST_PRECIPITATION,
ATTR_FORECAST_PRECIPITATION_PROBABILITY,
ATTR_FORECAST_UV_INDEX,
ATTR_FORECAST_WIND_BEARING,
)

DOMAIN = "metoffice"
Expand All @@ -33,22 +44,19 @@
METOFFICE_MONITORED_CONDITIONS = "metoffice_monitored_conditions"
METOFFICE_NAME = "metoffice_name"

MODE_3HOURLY = "3hourly"
MODE_DAILY = "daily"

CONDITION_CLASSES: dict[str, list[str]] = {
ATTR_CONDITION_CLEAR_NIGHT: ["0"],
ATTR_CONDITION_CLOUDY: ["7", "8"],
ATTR_CONDITION_FOG: ["5", "6"],
ATTR_CONDITION_HAIL: ["19", "20", "21"],
ATTR_CONDITION_LIGHTNING: ["30"],
ATTR_CONDITION_LIGHTNING_RAINY: ["28", "29"],
ATTR_CONDITION_PARTLYCLOUDY: ["2", "3"],
ATTR_CONDITION_POURING: ["13", "14", "15"],
ATTR_CONDITION_RAINY: ["9", "10", "11", "12"],
ATTR_CONDITION_SNOWY: ["22", "23", "24", "25", "26", "27"],
ATTR_CONDITION_SNOWY_RAINY: ["16", "17", "18"],
ATTR_CONDITION_SUNNY: ["1"],
CONDITION_CLASSES: dict[str, list[int]] = {
ATTR_CONDITION_CLEAR_NIGHT: [0],
ATTR_CONDITION_CLOUDY: [7, 8],
ATTR_CONDITION_FOG: [5, 6],
ATTR_CONDITION_HAIL: [19, 20, 21],
ATTR_CONDITION_LIGHTNING: [30],
ATTR_CONDITION_LIGHTNING_RAINY: [28, 29],
ATTR_CONDITION_PARTLYCLOUDY: [2, 3],
ATTR_CONDITION_POURING: [13, 14, 15],
ATTR_CONDITION_RAINY: [9, 10, 11, 12],
ATTR_CONDITION_SNOWY: [22, 23, 24, 25, 26, 27],
ATTR_CONDITION_SNOWY_RAINY: [16, 17, 18],
ATTR_CONDITION_SUNNY: [1],
ATTR_CONDITION_WINDY: [],
ATTR_CONDITION_WINDY_VARIANT: [],
ATTR_CONDITION_EXCEPTIONAL: [],
Expand All @@ -59,20 +67,28 @@
for cond_code in cond_codes
}

VISIBILITY_CLASSES = {
"VP": "Very Poor",
"PO": "Poor",
"MO": "Moderate",
"GO": "Good",
"VG": "Very Good",
"EX": "Excellent",
HOURLY_FORECAST_ATTRIBUTE_MAP: dict[str, str] = {
ATTR_FORECAST_CONDITION: "significantWeatherCode",
ATTR_FORECAST_NATIVE_APPARENT_TEMP: "feelsLikeTemperature",
ATTR_FORECAST_NATIVE_PRESSURE: "mslp",
ATTR_FORECAST_NATIVE_TEMP: "screenTemperature",
ATTR_FORECAST_PRECIPITATION: "totalPrecipAmount",
ATTR_FORECAST_PRECIPITATION_PROBABILITY: "probOfPrecipitation",
ATTR_FORECAST_UV_INDEX: "uvIndex",
ATTR_FORECAST_WIND_BEARING: "windDirectionFrom10m",
ATTR_FORECAST_NATIVE_WIND_SPEED: "windSpeed10m",
ATTR_FORECAST_NATIVE_WIND_GUST_SPEED: "windGustSpeed10m",
}

VISIBILITY_DISTANCE_CLASSES = {
"VP": "<1",
"PO": "1-4",
"MO": "4-10",
"GO": "10-20",
"VG": "20-40",
"EX": ">40",
DAILY_FORECAST_ATTRIBUTE_MAP: dict[str, str] = {
ATTR_FORECAST_CONDITION: "daySignificantWeatherCode",
ATTR_FORECAST_NATIVE_APPARENT_TEMP: "dayMaxFeelsLikeTemp",
ATTR_FORECAST_NATIVE_PRESSURE: "middayMslp",
ATTR_FORECAST_NATIVE_TEMP: "dayMaxScreenTemperature",
ATTR_FORECAST_NATIVE_TEMP_LOW: "nightMinScreenTemperature",
ATTR_FORECAST_PRECIPITATION_PROBABILITY: "dayProbabilityOfPrecipitation",
ATTR_FORECAST_UV_INDEX: "maxUvIndex",
ATTR_FORECAST_WIND_BEARING: "midday10MWindDirection",
ATTR_FORECAST_NATIVE_WIND_SPEED: "midday10MWindSpeed",
ATTR_FORECAST_NATIVE_WIND_GUST_SPEED: "midday10MWindGust",
}
18 changes: 0 additions & 18 deletions homeassistant/components/metoffice/data.py

This file was deleted.

Loading