2017-10-23 10:18:23 +00:00
|
|
|
"""
|
|
|
|
Integrate with DuckDNS.
|
|
|
|
|
|
|
|
For more details about this component, please refer to the documentation at
|
|
|
|
https://home-assistant.io/components/duckdns/
|
|
|
|
"""
|
2017-09-24 22:48:45 +00:00
|
|
|
from datetime import timedelta
|
|
|
|
import logging
|
|
|
|
|
|
|
|
import voluptuous as vol
|
|
|
|
|
|
|
|
from homeassistant.const import CONF_ACCESS_TOKEN, CONF_DOMAIN
|
|
|
|
import homeassistant.helpers.config_validation as cv
|
|
|
|
from homeassistant.helpers.event import async_track_time_interval
|
|
|
|
from homeassistant.helpers.aiohttp_client import async_get_clientsession
|
|
|
|
|
2017-10-23 10:18:23 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
ATTR_TXT = 'txt'
|
|
|
|
|
2017-09-24 22:48:45 +00:00
|
|
|
DOMAIN = 'duckdns'
|
2017-10-23 10:18:23 +00:00
|
|
|
|
2017-09-24 22:48:45 +00:00
|
|
|
INTERVAL = timedelta(minutes=5)
|
2017-10-23 10:18:23 +00:00
|
|
|
|
2017-09-24 22:48:45 +00:00
|
|
|
SERVICE_SET_TXT = 'set_txt'
|
2017-10-23 10:18:23 +00:00
|
|
|
|
|
|
|
UPDATE_URL = 'https://www.duckdns.org/update'
|
2017-09-24 22:48:45 +00:00
|
|
|
|
|
|
|
CONFIG_SCHEMA = vol.Schema({
|
|
|
|
DOMAIN: vol.Schema({
|
|
|
|
vol.Required(CONF_DOMAIN): cv.string,
|
|
|
|
vol.Required(CONF_ACCESS_TOKEN): cv.string,
|
|
|
|
})
|
|
|
|
}, extra=vol.ALLOW_EXTRA)
|
|
|
|
|
|
|
|
SERVICE_TXT_SCHEMA = vol.Schema({
|
|
|
|
vol.Required(ATTR_TXT): vol.Any(None, cv.string)
|
|
|
|
})
|
|
|
|
|
|
|
|
|
2018-10-01 06:52:42 +00:00
|
|
|
async def async_setup(hass, config):
|
2017-09-24 22:48:45 +00:00
|
|
|
"""Initialize the DuckDNS component."""
|
|
|
|
domain = config[DOMAIN][CONF_DOMAIN]
|
|
|
|
token = config[DOMAIN][CONF_ACCESS_TOKEN]
|
|
|
|
session = async_get_clientsession(hass)
|
|
|
|
|
2018-10-01 06:52:42 +00:00
|
|
|
result = await _update_duckdns(session, domain, token)
|
2017-09-24 22:48:45 +00:00
|
|
|
|
|
|
|
if not result:
|
|
|
|
return False
|
|
|
|
|
2018-10-01 06:52:42 +00:00
|
|
|
async def update_domain_interval(now):
|
2017-09-24 22:48:45 +00:00
|
|
|
"""Update the DuckDNS entry."""
|
2018-10-01 06:52:42 +00:00
|
|
|
await _update_duckdns(session, domain, token)
|
2017-09-24 22:48:45 +00:00
|
|
|
|
2018-10-01 06:52:42 +00:00
|
|
|
async def update_domain_service(call):
|
2017-09-24 22:48:45 +00:00
|
|
|
"""Update the DuckDNS entry."""
|
2018-10-01 06:52:42 +00:00
|
|
|
await _update_duckdns(
|
2017-10-23 10:18:23 +00:00
|
|
|
session, domain, token, txt=call.data[ATTR_TXT])
|
2017-09-24 22:48:45 +00:00
|
|
|
|
|
|
|
async_track_time_interval(hass, update_domain_interval, INTERVAL)
|
|
|
|
hass.services.async_register(
|
|
|
|
DOMAIN, SERVICE_SET_TXT, update_domain_service,
|
|
|
|
schema=SERVICE_TXT_SCHEMA)
|
|
|
|
|
|
|
|
return result
|
|
|
|
|
|
|
|
|
|
|
|
_SENTINEL = object()
|
|
|
|
|
|
|
|
|
2018-10-01 06:52:42 +00:00
|
|
|
async def _update_duckdns(session, domain, token, *, txt=_SENTINEL,
|
|
|
|
clear=False):
|
2017-09-24 22:48:45 +00:00
|
|
|
"""Update DuckDNS."""
|
|
|
|
params = {
|
|
|
|
'domains': domain,
|
|
|
|
'token': token,
|
|
|
|
}
|
|
|
|
|
|
|
|
if txt is not _SENTINEL:
|
|
|
|
if txt is None:
|
|
|
|
# Pass in empty txt value to indicate it's clearing txt record
|
|
|
|
params['txt'] = ''
|
|
|
|
clear = True
|
|
|
|
else:
|
|
|
|
params['txt'] = txt
|
|
|
|
|
|
|
|
if clear:
|
|
|
|
params['clear'] = 'true'
|
|
|
|
|
2018-10-01 06:52:42 +00:00
|
|
|
resp = await session.get(UPDATE_URL, params=params)
|
|
|
|
body = await resp.text()
|
2017-09-24 22:48:45 +00:00
|
|
|
|
|
|
|
if body != 'OK':
|
2017-10-23 10:18:23 +00:00
|
|
|
_LOGGER.warning("Updating DuckDNS domain failed: %s", domain)
|
2017-09-24 22:48:45 +00:00
|
|
|
return False
|
|
|
|
|
|
|
|
return True
|