146 lines
4.8 KiB
Python
146 lines
4.8 KiB
Python
"""Get WHOIS information for a given host."""
|
|
from __future__ import annotations
|
|
|
|
from datetime import timedelta
|
|
|
|
import voluptuous as vol
|
|
import whois
|
|
|
|
from homeassistant.components.sensor import PLATFORM_SCHEMA, SensorEntity
|
|
from homeassistant.config_entries import SOURCE_IMPORT, ConfigEntry
|
|
from homeassistant.const import CONF_DOMAIN, CONF_NAME, TIME_DAYS
|
|
from homeassistant.core import HomeAssistant
|
|
import homeassistant.helpers.config_validation as cv
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
|
from homeassistant.helpers.typing import ConfigType, DiscoveryInfoType
|
|
|
|
from .const import (
|
|
ATTR_EXPIRES,
|
|
ATTR_NAME_SERVERS,
|
|
ATTR_REGISTRAR,
|
|
ATTR_UPDATED,
|
|
DEFAULT_NAME,
|
|
DOMAIN,
|
|
LOGGER,
|
|
)
|
|
|
|
SCAN_INTERVAL = timedelta(hours=24)
|
|
|
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend(
|
|
{
|
|
vol.Required(CONF_DOMAIN): cv.string,
|
|
vol.Optional(CONF_NAME, default=DEFAULT_NAME): cv.string,
|
|
}
|
|
)
|
|
|
|
|
|
def setup_platform(
|
|
hass: HomeAssistant,
|
|
config: ConfigType,
|
|
add_entities: AddEntitiesCallback,
|
|
discovery_info: DiscoveryInfoType | None = None,
|
|
) -> None:
|
|
"""Set up the WHOIS sensor."""
|
|
LOGGER.warning(
|
|
"Configuration of the Whois platform in YAML is deprecated and will be "
|
|
"removed in Home Assistant 2022.4; Your existing configuration "
|
|
"has been imported into the UI automatically and can be safely removed "
|
|
"from your configuration.yaml file"
|
|
)
|
|
hass.async_create_task(
|
|
hass.config_entries.flow.async_init(
|
|
DOMAIN,
|
|
context={"source": SOURCE_IMPORT},
|
|
data={CONF_DOMAIN: config[CONF_DOMAIN], CONF_NAME: config[CONF_NAME]},
|
|
)
|
|
)
|
|
|
|
|
|
async def async_setup_entry(
|
|
hass: HomeAssistant,
|
|
entry: ConfigEntry,
|
|
async_add_entities: AddEntitiesCallback,
|
|
) -> None:
|
|
"""Set up the platform from config_entry."""
|
|
domain = entry.data[CONF_DOMAIN]
|
|
try:
|
|
info = await hass.async_add_executor_job(whois.whois, domain)
|
|
except whois.BaseException as ex: # pylint: disable=broad-except
|
|
LOGGER.error("Exception %s occurred during WHOIS lookup for %s", ex, domain)
|
|
return
|
|
|
|
if "expiration_date" not in info:
|
|
LOGGER.error("WHOIS lookup for %s didn't contain an expiration date", domain)
|
|
return
|
|
|
|
async_add_entities([WhoisSensor(domain)], True)
|
|
|
|
|
|
class WhoisSensor(SensorEntity):
|
|
"""Implementation of a WHOIS sensor."""
|
|
|
|
_attr_icon = "mdi:calendar-clock"
|
|
_attr_native_unit_of_measurement = TIME_DAYS
|
|
|
|
def __init__(self, domain: str) -> None:
|
|
"""Initialize the sensor."""
|
|
self._attr_name = domain
|
|
self.whois = whois.whois
|
|
self._domain = domain
|
|
|
|
def _empty_value_and_attributes(self) -> None:
|
|
"""Empty the state and attributes on an error."""
|
|
self._attr_native_value = None
|
|
self._attr_extra_state_attributes = {}
|
|
|
|
def update(self) -> None:
|
|
"""Get the current WHOIS data for the domain."""
|
|
try:
|
|
response = self.whois(self._domain)
|
|
except whois.BaseException as ex: # pylint: disable=broad-except
|
|
LOGGER.error("Exception %s occurred during WHOIS lookup", ex)
|
|
self._empty_value_and_attributes()
|
|
return
|
|
|
|
if response:
|
|
if "expiration_date" not in response:
|
|
LOGGER.error(
|
|
"Failed to find expiration_date in whois lookup response. "
|
|
"Did find: %s",
|
|
", ".join(response.keys()),
|
|
)
|
|
self._empty_value_and_attributes()
|
|
return
|
|
|
|
if not response["expiration_date"]:
|
|
LOGGER.error("Whois response contains empty expiration_date")
|
|
self._empty_value_and_attributes()
|
|
return
|
|
|
|
attrs = {}
|
|
|
|
expiration_date = response["expiration_date"]
|
|
if isinstance(expiration_date, list):
|
|
attrs[ATTR_EXPIRES] = expiration_date[0].isoformat()
|
|
expiration_date = expiration_date[0]
|
|
else:
|
|
attrs[ATTR_EXPIRES] = expiration_date.isoformat()
|
|
|
|
if "nameservers" in response:
|
|
attrs[ATTR_NAME_SERVERS] = " ".join(response["nameservers"])
|
|
|
|
if "updated_date" in response:
|
|
update_date = response["updated_date"]
|
|
if isinstance(update_date, list):
|
|
attrs[ATTR_UPDATED] = update_date[0].isoformat()
|
|
else:
|
|
attrs[ATTR_UPDATED] = update_date.isoformat()
|
|
|
|
if "registrar" in response:
|
|
attrs[ATTR_REGISTRAR] = response["registrar"]
|
|
|
|
time_delta = expiration_date - expiration_date.now()
|
|
|
|
self._attr_extra_state_attributes = attrs
|
|
self._attr_native_value = time_delta.days
|