mirror of
https://github.com/home-assistant/core.git
synced 2025-07-20 19:57:07 +00:00
Added DWD WarnApp Sensor (#8657)
* Added DWD WarnApp Sensor * Fixed some idents and spaces * Removed unused imports * Removed comment * Some fixes * Added throttle * Renamed sensor to dwd weather warnings * Renamed test file * shorten lines * shorten lines * Implemented changes requested by fabaff * added ATTRIBUTION * move ATTRIBUTION to existing method * fixed lint tests * Fix linter issues * Fix linter issues * Fix linter * Fixed linter
This commit is contained in:
parent
0b1677de6d
commit
9ede0f57e6
@ -451,6 +451,7 @@ omit =
|
|||||||
homeassistant/components/sensor/dovado.py
|
homeassistant/components/sensor/dovado.py
|
||||||
homeassistant/components/sensor/dte_energy_bridge.py
|
homeassistant/components/sensor/dte_energy_bridge.py
|
||||||
homeassistant/components/sensor/dublin_bus_transport.py
|
homeassistant/components/sensor/dublin_bus_transport.py
|
||||||
|
homeassistant/components/sensor/dwd_weather_warnings.py
|
||||||
homeassistant/components/sensor/ebox.py
|
homeassistant/components/sensor/ebox.py
|
||||||
homeassistant/components/sensor/eddystone_temperature.py
|
homeassistant/components/sensor/eddystone_temperature.py
|
||||||
homeassistant/components/sensor/eliqonline.py
|
homeassistant/components/sensor/eliqonline.py
|
||||||
|
243
homeassistant/components/sensor/dwd_weather_warnings.py
Normal file
243
homeassistant/components/sensor/dwd_weather_warnings.py
Normal file
@ -0,0 +1,243 @@
|
|||||||
|
"""
|
||||||
|
Support for getting statistical data from a DWD Weather Warnings.
|
||||||
|
|
||||||
|
For more details about this platform, please refer to the documentation at
|
||||||
|
https://home-assistant.io/components/sensor.dwd_weather_warnings/
|
||||||
|
|
||||||
|
Data is fetched from DWD:
|
||||||
|
https://rcccm.dwd.de/DE/wetter/warnungen_aktuell/objekt_einbindung/objekteinbindung.html
|
||||||
|
|
||||||
|
Warnungen vor extremem Unwetter (Stufe 4)
|
||||||
|
Unwetterwarnungen (Stufe 3)
|
||||||
|
Warnungen vor markantem Wetter (Stufe 2)
|
||||||
|
Wetterwarnungen (Stufe 1)
|
||||||
|
"""
|
||||||
|
import logging
|
||||||
|
import json
|
||||||
|
from datetime import timedelta
|
||||||
|
|
||||||
|
import voluptuous as vol
|
||||||
|
|
||||||
|
import homeassistant.helpers.config_validation as cv
|
||||||
|
from homeassistant.helpers.entity import Entity
|
||||||
|
from homeassistant.components.sensor import PLATFORM_SCHEMA
|
||||||
|
from homeassistant.const import (
|
||||||
|
ATTR_ATTRIBUTION, CONF_NAME, CONF_MONITORED_CONDITIONS)
|
||||||
|
from homeassistant.util import Throttle
|
||||||
|
import homeassistant.util.dt as dt_util
|
||||||
|
from homeassistant.components.sensor.rest import RestData
|
||||||
|
|
||||||
|
_LOGGER = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
ATTRIBUTION = "Data provided by DWD"
|
||||||
|
|
||||||
|
DEFAULT_NAME = 'DWD-Weather-Warnings'
|
||||||
|
|
||||||
|
CONF_REGION_NAME = 'region_name'
|
||||||
|
|
||||||
|
SCAN_INTERVAL = timedelta(minutes=15)
|
||||||
|
|
||||||
|
MONITORED_CONDITIONS = {
|
||||||
|
'current_warning_level': ['Current Warning Level',
|
||||||
|
None, 'mdi:close-octagon-outline'],
|
||||||
|
'advance_warning_level': ['Advance Warning Level',
|
||||||
|
None, 'mdi:close-octagon-outline'],
|
||||||
|
}
|
||||||
|
|
||||||
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend({
|
||||||
|
vol.Optional(CONF_REGION_NAME): cv.string,
|
||||||
|
vol.Optional(CONF_NAME, default=DEFAULT_NAME): cv.string,
|
||||||
|
vol.Optional(CONF_MONITORED_CONDITIONS, default=MONITORED_CONDITIONS):
|
||||||
|
vol.All(cv.ensure_list, [vol.In(MONITORED_CONDITIONS)]),
|
||||||
|
})
|
||||||
|
|
||||||
|
|
||||||
|
def setup_platform(hass, config, add_devices, discovery_info=None):
|
||||||
|
"""Set up the DWD-Weather-Warnings sensor."""
|
||||||
|
name = config.get(CONF_NAME)
|
||||||
|
region_name = config.get(CONF_REGION_NAME)
|
||||||
|
|
||||||
|
api = DwdWeatherWarningsAPI(region_name)
|
||||||
|
|
||||||
|
sensors = [DwdWeatherWarningsSensor(api, name, condition)
|
||||||
|
for condition in config[CONF_MONITORED_CONDITIONS]]
|
||||||
|
|
||||||
|
add_devices(sensors, True)
|
||||||
|
|
||||||
|
|
||||||
|
class DwdWeatherWarningsSensor(Entity):
|
||||||
|
"""Representation of a DWD-Weather-Warnings sensor."""
|
||||||
|
|
||||||
|
def __init__(self, api, name, variable):
|
||||||
|
"""Initialize a DWD-Weather-Warnings sensor."""
|
||||||
|
self._api = api
|
||||||
|
self._name = name
|
||||||
|
self._var_id = variable
|
||||||
|
|
||||||
|
variable_info = MONITORED_CONDITIONS[variable]
|
||||||
|
self._var_name = variable_info[0]
|
||||||
|
self._var_units = variable_info[1]
|
||||||
|
self._var_icon = variable_info[2]
|
||||||
|
|
||||||
|
@property
|
||||||
|
def name(self):
|
||||||
|
"""Return the name of the sensor."""
|
||||||
|
return "{} {}".format(self._name, self._var_name)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def icon(self):
|
||||||
|
"""Icon to use in the frontend, if any."""
|
||||||
|
return self._var_icon
|
||||||
|
|
||||||
|
@property
|
||||||
|
def unit_of_measurement(self):
|
||||||
|
"""Return the unit the value is expressed in."""
|
||||||
|
return self._var_units
|
||||||
|
|
||||||
|
# pylint: disable=no-member
|
||||||
|
@property
|
||||||
|
def state(self):
|
||||||
|
"""Return the state of the device."""
|
||||||
|
try:
|
||||||
|
return round(self._api.data[self._var_id], 2)
|
||||||
|
except TypeError:
|
||||||
|
return self._api.data[self._var_id]
|
||||||
|
|
||||||
|
# pylint: disable=no-member
|
||||||
|
@property
|
||||||
|
def device_state_attributes(self):
|
||||||
|
"""Return the state attributes of the DWD-Weather-Warnings."""
|
||||||
|
data = {
|
||||||
|
ATTR_ATTRIBUTION: ATTRIBUTION,
|
||||||
|
'region_name': self._api.region_name
|
||||||
|
}
|
||||||
|
|
||||||
|
if self._api.region_id is not None:
|
||||||
|
data['region_id'] = self._api.region_id
|
||||||
|
|
||||||
|
if self._api.region_state is not None:
|
||||||
|
data['region_state'] = self._api.region_state
|
||||||
|
|
||||||
|
if self._api.data['time'] is not None:
|
||||||
|
data['last_update'] = dt_util.as_local(
|
||||||
|
dt_util.utc_from_timestamp(self._api.data['time'] / 1000))
|
||||||
|
|
||||||
|
if self._var_id == 'current_warning_level':
|
||||||
|
prefix = 'current'
|
||||||
|
elif self._var_id == 'advance_warning_level':
|
||||||
|
prefix = 'advance'
|
||||||
|
else:
|
||||||
|
raise Exception('Unknown warning type')
|
||||||
|
|
||||||
|
data['warning_count'] = self._api.data[prefix + '_warning_count']
|
||||||
|
i = 0
|
||||||
|
for event in self._api.data[prefix + '_warnings']:
|
||||||
|
i = i + 1
|
||||||
|
|
||||||
|
data['warning_{}_name'.format(i)] = event['event']
|
||||||
|
data['warning_{}_level'.format(i)] = event['level']
|
||||||
|
data['warning_{}_type'.format(i)] = event['type']
|
||||||
|
if len(event['headline']) > 0:
|
||||||
|
data['warning_{}_headline'.format(i)] = event['headline']
|
||||||
|
if len(event['description']) > 0:
|
||||||
|
data['warning_{}_description'.format(i)] = event['description']
|
||||||
|
if len(event['instruction']) > 0:
|
||||||
|
data['warning_{}_instruction'.format(i)] = event['instruction']
|
||||||
|
|
||||||
|
if event['start'] is not None:
|
||||||
|
data['warning_{}_start'.format(i)] = dt_util.as_local(
|
||||||
|
dt_util.utc_from_timestamp(event['start'] / 1000))
|
||||||
|
|
||||||
|
if event['end'] is not None:
|
||||||
|
data['warning_{}_end'.format(i)] = dt_util.as_local(
|
||||||
|
dt_util.utc_from_timestamp(event['end'] / 1000))
|
||||||
|
|
||||||
|
return data
|
||||||
|
|
||||||
|
@property
|
||||||
|
def available(self):
|
||||||
|
"""Could the device be accessed during the last update call."""
|
||||||
|
return self._api.available
|
||||||
|
|
||||||
|
def update(self):
|
||||||
|
"""Get the latest data from the DWD-Weather-Warnings API."""
|
||||||
|
self._api.update()
|
||||||
|
|
||||||
|
|
||||||
|
class DwdWeatherWarningsAPI(object):
|
||||||
|
"""Get the latest data and update the states."""
|
||||||
|
|
||||||
|
def __init__(self, region_name):
|
||||||
|
"""Initialize the data object."""
|
||||||
|
resource = "{}{}{}?{}".format(
|
||||||
|
'https://',
|
||||||
|
'www.dwd.de',
|
||||||
|
'/DWD/warnungen/warnapp_landkreise/json/warnings.json',
|
||||||
|
'jsonp=loadWarnings'
|
||||||
|
)
|
||||||
|
|
||||||
|
self._rest = RestData('GET', resource, None, None, None, True)
|
||||||
|
self.region_name = region_name
|
||||||
|
self.region_id = None
|
||||||
|
self.region_state = None
|
||||||
|
self.data = None
|
||||||
|
self.available = True
|
||||||
|
self.update()
|
||||||
|
|
||||||
|
@Throttle(SCAN_INTERVAL)
|
||||||
|
def update(self):
|
||||||
|
"""Get the latest data from the DWD-Weather-Warnings."""
|
||||||
|
try:
|
||||||
|
self._rest.update()
|
||||||
|
|
||||||
|
json_string = self._rest.data[24:len(self._rest.data) - 2]
|
||||||
|
json_obj = json.loads(json_string)
|
||||||
|
|
||||||
|
data = {'time': json_obj['time']}
|
||||||
|
|
||||||
|
for mykey, myvalue in {
|
||||||
|
'current': 'warnings',
|
||||||
|
'advance': 'vorabInformation'
|
||||||
|
}.items():
|
||||||
|
|
||||||
|
_LOGGER.debug("Found %d %s global DWD warnings",
|
||||||
|
len(json_obj[myvalue]), mykey)
|
||||||
|
|
||||||
|
data['{}_warning_level'.format(mykey)] = 0
|
||||||
|
my_warnings = []
|
||||||
|
|
||||||
|
if self.region_id is not None:
|
||||||
|
# get a specific region_id
|
||||||
|
if self.region_id in json_obj[myvalue]:
|
||||||
|
my_warnings = json_obj[myvalue][self.region_id]
|
||||||
|
|
||||||
|
else:
|
||||||
|
# loop through all items to find warnings, region_id
|
||||||
|
# and region_state for region_name
|
||||||
|
for key in json_obj[myvalue]:
|
||||||
|
my_region = json_obj[myvalue][key][0]['regionName']
|
||||||
|
if my_region != self.region_name:
|
||||||
|
continue
|
||||||
|
my_warnings = json_obj[myvalue][key]
|
||||||
|
my_state = json_obj[myvalue][key][0]['stateShort']
|
||||||
|
self.region_id = key
|
||||||
|
self.region_state = my_state
|
||||||
|
break
|
||||||
|
|
||||||
|
# Get max warning level
|
||||||
|
maxlevel = data['{}_warning_level'.format(mykey)]
|
||||||
|
for event in my_warnings:
|
||||||
|
if event['level'] >= maxlevel:
|
||||||
|
data['{}_warning_level'.format(mykey)] = event['level']
|
||||||
|
|
||||||
|
data['{}_warning_count'.format(mykey)] = len(my_warnings)
|
||||||
|
data['{}_warnings'.format(mykey)] = my_warnings
|
||||||
|
|
||||||
|
_LOGGER.debug("Found %d %s local DWD warnings",
|
||||||
|
len(my_warnings), mykey)
|
||||||
|
|
||||||
|
self.data = data
|
||||||
|
self.available = True
|
||||||
|
except TypeError:
|
||||||
|
_LOGGER.error("Unable to fetch data from DWD-Weather-Warnings")
|
||||||
|
self.available = False
|
Loading…
x
Reference in New Issue
Block a user