core/homeassistant/components/environment_canada/sensor.py

163 lines
4.8 KiB
Python
Raw Normal View History

"""
Support for the Environment Canada weather service.
For more details about this platform, please refer to the documentation at
https://home-assistant.io/components/sensor.environment_canada/
"""
from datetime import datetime, timedelta
import logging
import re
import voluptuous as vol
from homeassistant.components.sensor import PLATFORM_SCHEMA
from homeassistant.const import (
2019-07-31 19:25:30 +00:00
TEMP_CELSIUS,
CONF_LATITUDE,
CONF_LONGITUDE,
ATTR_ATTRIBUTION,
ATTR_LOCATION,
)
from homeassistant.helpers.entity import Entity
import homeassistant.helpers.config_validation as cv
_LOGGER = logging.getLogger(__name__)
SCAN_INTERVAL = timedelta(minutes=10)
2019-07-31 19:25:30 +00:00
ATTR_UPDATED = "updated"
ATTR_STATION = "station"
ATTR_DETAIL = "alert detail"
ATTR_TIME = "alert time"
CONF_ATTRIBUTION = "Data provided by Environment Canada"
2019-07-31 19:25:30 +00:00
CONF_STATION = "station"
CONF_LANGUAGE = "language"
def validate_station(station):
"""Check that the station ID is well-formed."""
if station is None:
return
2019-07-31 19:25:30 +00:00
if not re.fullmatch(r"[A-Z]{2}/s0000\d{3}", station):
raise vol.error.Invalid('Station ID must be of the form "XX/s0000###"')
return station
2019-07-31 19:25:30 +00:00
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend(
{
vol.Required(CONF_LANGUAGE, default="english"): vol.In(["english", "french"]),
vol.Optional(CONF_STATION): validate_station,
vol.Inclusive(CONF_LATITUDE, "latlon"): cv.latitude,
vol.Inclusive(CONF_LONGITUDE, "latlon"): cv.longitude,
}
)
def setup_platform(hass, config, add_entities, discovery_info=None):
"""Set up the Environment Canada sensor."""
from env_canada import ECData
if config.get(CONF_STATION):
2019-07-31 19:25:30 +00:00
ec_data = ECData(
station_id=config[CONF_STATION], language=config.get(CONF_LANGUAGE)
)
else:
lat = config.get(CONF_LATITUDE, hass.config.latitude)
lon = config.get(CONF_LONGITUDE, hass.config.longitude)
2019-07-31 19:25:30 +00:00
ec_data = ECData(coordinates=(lat, lon), language=config.get(CONF_LANGUAGE))
sensor_list = list(ec_data.conditions.keys()) + list(ec_data.alerts.keys())
2019-07-31 19:25:30 +00:00
sensor_list.remove("icon_code")
add_entities([ECSensor(sensor_type, ec_data) for sensor_type in sensor_list], True)
class ECSensor(Entity):
"""Implementation of an Environment Canada sensor."""
def __init__(self, sensor_type, ec_data):
"""Initialize the sensor."""
self.sensor_type = sensor_type
self.ec_data = ec_data
self._unique_id = None
self._name = None
self._state = None
self._attr = None
self._unit = None
@property
def unique_id(self) -> str:
"""Return the unique ID of the sensor."""
return self._unique_id
@property
def name(self):
"""Return the name of the sensor."""
return self._name
@property
def state(self):
"""Return the state of the sensor."""
return self._state
@property
def device_state_attributes(self):
"""Return the state attributes of the device."""
return self._attr
@property
def unit_of_measurement(self):
"""Return the units of measurement."""
return self._unit
def update(self):
"""Update current conditions."""
self.ec_data.update()
self.ec_data.conditions.update(self.ec_data.alerts)
conditions = self.ec_data.conditions
metadata = self.ec_data.metadata
sensor_data = conditions.get(self.sensor_type)
2019-07-31 19:25:30 +00:00
self._unique_id = "{}-{}".format(metadata["location"], self.sensor_type)
self._attr = {}
2019-07-31 19:25:30 +00:00
self._name = sensor_data.get("label")
value = sensor_data.get("value")
if isinstance(value, list):
2019-07-31 19:25:30 +00:00
self._state = " | ".join([str(s.get("title")) for s in value])
self._attr.update(
{
ATTR_DETAIL: " | ".join([str(s.get("detail")) for s in value]),
ATTR_TIME: " | ".join([str(s.get("date")) for s in value]),
}
)
elif self.sensor_type == "tendency":
self._state = str(value).capitalize()
else:
self._state = value
if sensor_data.get("unit") == "C" or self.sensor_type in [
"wind_chill",
"humidex",
]:
self._unit = TEMP_CELSIUS
else:
2019-07-31 19:25:30 +00:00
self._unit = sensor_data.get("unit")
2019-07-31 19:25:30 +00:00
timestamp = metadata.get("timestamp")
if timestamp:
2019-07-31 19:25:30 +00:00
updated_utc = datetime.strptime(timestamp, "%Y%m%d%H%M%S").isoformat()
else:
updated_utc = None
2019-07-31 19:25:30 +00:00
self._attr.update(
{
ATTR_ATTRIBUTION: CONF_ATTRIBUTION,
ATTR_UPDATED: updated_utc,
ATTR_LOCATION: metadata.get("location"),
ATTR_STATION: metadata.get("station"),
}
)