2019-04-03 15:40:03 +00:00
|
|
|
"""Sensor for checking the air quality around Norway."""
|
2022-01-03 11:07:05 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2019-01-27 20:32:23 +00:00
|
|
|
from datetime import timedelta
|
|
|
|
import logging
|
|
|
|
|
2019-10-18 23:05:36 +00:00
|
|
|
from niluclient import (
|
|
|
|
CO,
|
|
|
|
CO2,
|
|
|
|
NO,
|
|
|
|
NO2,
|
|
|
|
NOX,
|
|
|
|
OZONE,
|
|
|
|
PM1,
|
|
|
|
PM10,
|
|
|
|
PM25,
|
|
|
|
POLLUTION_INDEX,
|
|
|
|
SO2,
|
|
|
|
create_location_client,
|
|
|
|
create_station_client,
|
|
|
|
lookup_stations_in_area,
|
|
|
|
)
|
2019-01-27 20:32:23 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
from homeassistant.components.air_quality import PLATFORM_SCHEMA, AirQualityEntity
|
2019-01-27 20:32:23 +00:00
|
|
|
from homeassistant.const import (
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_LATITUDE,
|
|
|
|
CONF_LONGITUDE,
|
|
|
|
CONF_NAME,
|
|
|
|
CONF_SHOW_ON_MAP,
|
|
|
|
)
|
2022-01-03 11:07:05 +00:00
|
|
|
from homeassistant.core import HomeAssistant
|
2019-01-27 20:32:23 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
2022-01-03 11:07:05 +00:00
|
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
|
|
|
from homeassistant.helpers.typing import ConfigType, DiscoveryInfoType
|
2019-01-27 20:32:23 +00:00
|
|
|
from homeassistant.util import Throttle
|
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_AREA = "area"
|
|
|
|
ATTR_POLLUTION_INDEX = "nilu_pollution_index"
|
2019-01-27 20:32:23 +00:00
|
|
|
ATTRIBUTION = "Data provided by luftkvalitet.info and nilu.no"
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_AREA = "area"
|
|
|
|
CONF_STATION = "stations"
|
2019-01-27 20:32:23 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
DEFAULT_NAME = "NILU"
|
2019-01-27 20:32:23 +00:00
|
|
|
|
|
|
|
SCAN_INTERVAL = timedelta(minutes=30)
|
|
|
|
|
|
|
|
CONF_ALLOWED_AREAS = [
|
2019-07-31 19:25:30 +00:00
|
|
|
"Bergen",
|
|
|
|
"Birkenes",
|
|
|
|
"Bodø",
|
|
|
|
"Brumunddal",
|
|
|
|
"Bærum",
|
|
|
|
"Drammen",
|
|
|
|
"Elverum",
|
|
|
|
"Fredrikstad",
|
|
|
|
"Gjøvik",
|
|
|
|
"Grenland",
|
|
|
|
"Halden",
|
|
|
|
"Hamar",
|
|
|
|
"Harstad",
|
|
|
|
"Hurdal",
|
|
|
|
"Karasjok",
|
|
|
|
"Kristiansand",
|
|
|
|
"Kårvatn",
|
|
|
|
"Lillehammer",
|
|
|
|
"Lillesand",
|
|
|
|
"Lillestrøm",
|
|
|
|
"Lørenskog",
|
|
|
|
"Mo i Rana",
|
|
|
|
"Moss",
|
|
|
|
"Narvik",
|
|
|
|
"Oslo",
|
|
|
|
"Prestebakke",
|
|
|
|
"Sandve",
|
|
|
|
"Sarpsborg",
|
|
|
|
"Stavanger",
|
|
|
|
"Sør-Varanger",
|
|
|
|
"Tromsø",
|
|
|
|
"Trondheim",
|
|
|
|
"Tustervatn",
|
|
|
|
"Zeppelinfjellet",
|
|
|
|
"Ålesund",
|
2019-01-27 20:32:23 +00:00
|
|
|
]
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend(
|
|
|
|
{
|
|
|
|
vol.Inclusive(
|
|
|
|
CONF_LATITUDE, "coordinates", "Latitude and longitude must exist together"
|
|
|
|
): cv.latitude,
|
|
|
|
vol.Inclusive(
|
|
|
|
CONF_LONGITUDE, "coordinates", "Latitude and longitude must exist together"
|
|
|
|
): cv.longitude,
|
|
|
|
vol.Exclusive(
|
|
|
|
CONF_AREA,
|
|
|
|
"station_collection",
|
2022-12-22 12:35:47 +00:00
|
|
|
(
|
|
|
|
"Can only configure one specific station or "
|
|
|
|
"stations in a specific area pr sensor. "
|
|
|
|
"Please only configure station or area."
|
|
|
|
),
|
2019-07-31 19:25:30 +00:00
|
|
|
): vol.All(cv.string, vol.In(CONF_ALLOWED_AREAS)),
|
|
|
|
vol.Exclusive(
|
|
|
|
CONF_STATION,
|
|
|
|
"station_collection",
|
2022-12-22 12:35:47 +00:00
|
|
|
(
|
|
|
|
"Can only configure one specific station or "
|
|
|
|
"stations in a specific area pr sensor. "
|
|
|
|
"Please only configure station or area."
|
|
|
|
),
|
2019-07-31 19:25:30 +00:00
|
|
|
): vol.All(cv.ensure_list, [cv.string]),
|
|
|
|
vol.Optional(CONF_NAME, default=DEFAULT_NAME): cv.string,
|
|
|
|
vol.Optional(CONF_SHOW_ON_MAP, default=False): cv.boolean,
|
|
|
|
}
|
|
|
|
)
|
2019-01-27 20:32:23 +00:00
|
|
|
|
|
|
|
|
2022-01-03 11:07:05 +00:00
|
|
|
def setup_platform(
|
|
|
|
hass: HomeAssistant,
|
|
|
|
config: ConfigType,
|
|
|
|
add_entities: AddEntitiesCallback,
|
|
|
|
discovery_info: DiscoveryInfoType | None = None,
|
|
|
|
) -> None:
|
2019-01-27 20:32:23 +00:00
|
|
|
"""Set up the NILU air quality sensor."""
|
2022-07-09 18:01:39 +00:00
|
|
|
name: str = config[CONF_NAME]
|
|
|
|
area: str | None = config.get(CONF_AREA)
|
|
|
|
stations: list[str] | None = config.get(CONF_STATION)
|
|
|
|
show_on_map: bool = config[CONF_SHOW_ON_MAP]
|
2019-01-27 20:32:23 +00:00
|
|
|
|
|
|
|
sensors = []
|
|
|
|
|
|
|
|
if area:
|
2019-10-18 23:05:36 +00:00
|
|
|
stations = lookup_stations_in_area(area)
|
2022-07-09 18:01:39 +00:00
|
|
|
elif not stations:
|
2019-01-27 20:32:23 +00:00
|
|
|
latitude = config.get(CONF_LATITUDE, hass.config.latitude)
|
|
|
|
longitude = config.get(CONF_LONGITUDE, hass.config.longitude)
|
2019-10-18 23:05:36 +00:00
|
|
|
location_client = create_location_client(latitude, longitude)
|
2019-01-27 20:32:23 +00:00
|
|
|
stations = location_client.station_names
|
|
|
|
|
2022-07-09 18:01:39 +00:00
|
|
|
assert stations is not None
|
2019-01-27 20:32:23 +00:00
|
|
|
for station in stations:
|
2019-10-18 23:05:36 +00:00
|
|
|
client = NiluData(create_station_client(station))
|
2019-01-27 20:32:23 +00:00
|
|
|
client.update()
|
|
|
|
if client.data.sensors:
|
|
|
|
sensors.append(NiluSensor(client, name, show_on_map))
|
|
|
|
else:
|
|
|
|
_LOGGER.warning("%s didn't give any sensors results", station)
|
|
|
|
|
|
|
|
add_entities(sensors, True)
|
|
|
|
|
|
|
|
|
|
|
|
class NiluData:
|
|
|
|
"""Class for handling the data retrieval."""
|
|
|
|
|
|
|
|
def __init__(self, api):
|
|
|
|
"""Initialize the data object."""
|
|
|
|
self.api = api
|
|
|
|
|
|
|
|
@property
|
|
|
|
def data(self):
|
|
|
|
"""Get data cached in client."""
|
|
|
|
return self.api.data
|
|
|
|
|
|
|
|
@Throttle(SCAN_INTERVAL)
|
|
|
|
def update(self):
|
|
|
|
"""Get the latest data from nilu API."""
|
|
|
|
self.api.update()
|
|
|
|
|
|
|
|
|
|
|
|
class NiluSensor(AirQualityEntity):
|
|
|
|
"""Single nilu station air sensor."""
|
|
|
|
|
2021-05-20 12:06:44 +00:00
|
|
|
def __init__(self, api_data: NiluData, name: str, show_on_map: bool) -> None:
|
2019-01-27 20:32:23 +00:00
|
|
|
"""Initialize the sensor."""
|
|
|
|
self._api = api_data
|
2019-09-03 18:35:00 +00:00
|
|
|
self._name = f"{name} {api_data.data.name}"
|
2019-01-27 20:32:23 +00:00
|
|
|
self._max_aqi = None
|
|
|
|
self._attrs = {}
|
|
|
|
|
|
|
|
if show_on_map:
|
|
|
|
self._attrs[CONF_LATITUDE] = api_data.data.latitude
|
|
|
|
self._attrs[CONF_LONGITUDE] = api_data.data.longitude
|
|
|
|
|
|
|
|
@property
|
|
|
|
def attribution(self) -> str:
|
|
|
|
"""Return the attribution."""
|
|
|
|
return ATTRIBUTION
|
|
|
|
|
|
|
|
@property
|
2021-03-11 19:11:25 +00:00
|
|
|
def extra_state_attributes(self) -> dict:
|
2019-01-27 20:32:23 +00:00
|
|
|
"""Return other details about the sensor state."""
|
|
|
|
return self._attrs
|
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self) -> str:
|
|
|
|
"""Return the name of the sensor."""
|
|
|
|
return self._name
|
|
|
|
|
|
|
|
@property
|
2022-07-09 18:01:39 +00:00
|
|
|
def air_quality_index(self) -> str | None:
|
2019-01-27 20:32:23 +00:00
|
|
|
"""Return the Air Quality Index (AQI)."""
|
|
|
|
return self._max_aqi
|
|
|
|
|
|
|
|
@property
|
2022-07-09 18:01:39 +00:00
|
|
|
def carbon_monoxide(self) -> str | None:
|
2019-01-27 20:32:23 +00:00
|
|
|
"""Return the CO (carbon monoxide) level."""
|
|
|
|
return self.get_component_state(CO)
|
|
|
|
|
|
|
|
@property
|
2022-07-09 18:01:39 +00:00
|
|
|
def carbon_dioxide(self) -> str | None:
|
2019-01-27 20:32:23 +00:00
|
|
|
"""Return the CO2 (carbon dioxide) level."""
|
|
|
|
return self.get_component_state(CO2)
|
|
|
|
|
|
|
|
@property
|
2022-07-09 18:01:39 +00:00
|
|
|
def nitrogen_oxide(self) -> str | None:
|
2019-01-27 20:32:23 +00:00
|
|
|
"""Return the N2O (nitrogen oxide) level."""
|
|
|
|
return self.get_component_state(NOX)
|
|
|
|
|
|
|
|
@property
|
2022-07-09 18:01:39 +00:00
|
|
|
def nitrogen_monoxide(self) -> str | None:
|
2019-01-27 20:32:23 +00:00
|
|
|
"""Return the NO (nitrogen monoxide) level."""
|
|
|
|
return self.get_component_state(NO)
|
|
|
|
|
|
|
|
@property
|
2022-07-09 18:01:39 +00:00
|
|
|
def nitrogen_dioxide(self) -> str | None:
|
2019-01-27 20:32:23 +00:00
|
|
|
"""Return the NO2 (nitrogen dioxide) level."""
|
|
|
|
return self.get_component_state(NO2)
|
|
|
|
|
|
|
|
@property
|
2022-07-09 18:01:39 +00:00
|
|
|
def ozone(self) -> str | None:
|
2019-01-27 20:32:23 +00:00
|
|
|
"""Return the O3 (ozone) level."""
|
|
|
|
return self.get_component_state(OZONE)
|
|
|
|
|
|
|
|
@property
|
2022-07-09 18:01:39 +00:00
|
|
|
def particulate_matter_2_5(self) -> str | None:
|
2019-01-27 20:32:23 +00:00
|
|
|
"""Return the particulate matter 2.5 level."""
|
|
|
|
return self.get_component_state(PM25)
|
|
|
|
|
|
|
|
@property
|
2022-07-09 18:01:39 +00:00
|
|
|
def particulate_matter_10(self) -> str | None:
|
2019-01-27 20:32:23 +00:00
|
|
|
"""Return the particulate matter 10 level."""
|
|
|
|
return self.get_component_state(PM10)
|
|
|
|
|
|
|
|
@property
|
2022-07-09 18:01:39 +00:00
|
|
|
def particulate_matter_0_1(self) -> str | None:
|
2019-01-27 20:32:23 +00:00
|
|
|
"""Return the particulate matter 0.1 level."""
|
|
|
|
return self.get_component_state(PM1)
|
|
|
|
|
|
|
|
@property
|
2022-07-09 18:01:39 +00:00
|
|
|
def sulphur_dioxide(self) -> str | None:
|
2019-01-27 20:32:23 +00:00
|
|
|
"""Return the SO2 (sulphur dioxide) level."""
|
|
|
|
return self.get_component_state(SO2)
|
|
|
|
|
2022-07-09 18:01:39 +00:00
|
|
|
def get_component_state(self, component_name: str) -> str | None:
|
2019-01-27 20:32:23 +00:00
|
|
|
"""Return formatted value of specified component."""
|
|
|
|
if component_name in self._api.data.sensors:
|
|
|
|
sensor = self._api.data.sensors[component_name]
|
|
|
|
return sensor.value
|
|
|
|
return None
|
|
|
|
|
|
|
|
def update(self) -> None:
|
|
|
|
"""Update the sensor."""
|
|
|
|
self._api.update()
|
|
|
|
|
|
|
|
sensors = self._api.data.sensors.values()
|
|
|
|
if sensors:
|
2021-07-19 08:46:09 +00:00
|
|
|
max_index = max(s.pollution_index for s in sensors)
|
2019-01-27 20:32:23 +00:00
|
|
|
self._max_aqi = max_index
|
2019-10-18 23:05:36 +00:00
|
|
|
self._attrs[ATTR_POLLUTION_INDEX] = POLLUTION_INDEX[self._max_aqi]
|
2019-01-27 20:32:23 +00:00
|
|
|
|
|
|
|
self._attrs[ATTR_AREA] = self._api.data.area
|