2019-02-13 20:21:14 +00:00
|
|
|
"""Platform for the Daikin AC."""
|
2018-12-16 15:19:18 +00:00
|
|
|
import asyncio
|
2018-01-04 10:05:27 +00:00
|
|
|
from datetime import timedelta
|
2018-12-16 15:19:18 +00:00
|
|
|
import logging
|
2018-01-04 10:05:27 +00:00
|
|
|
|
2019-04-08 17:08:03 +00:00
|
|
|
from aiohttp import ClientConnectionError
|
|
|
|
from async_timeout import timeout
|
2020-05-06 11:59:07 +00:00
|
|
|
from pydaikin.daikin_base import Appliance
|
2018-01-04 10:05:27 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
2019-01-07 12:04:53 +00:00
|
|
|
from homeassistant.config_entries import SOURCE_IMPORT, ConfigEntry
|
2020-05-06 11:59:07 +00:00
|
|
|
from homeassistant.const import CONF_HOST, CONF_HOSTS, CONF_PASSWORD
|
2019-04-10 07:44:00 +00:00
|
|
|
from homeassistant.exceptions import ConfigEntryNotReady
|
2018-01-04 10:05:27 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
2018-12-16 15:19:18 +00:00
|
|
|
from homeassistant.helpers.typing import HomeAssistantType
|
2018-01-04 10:05:27 +00:00
|
|
|
from homeassistant.util import Throttle
|
|
|
|
|
2019-11-16 09:22:07 +00:00
|
|
|
from . import config_flow # noqa: F401
|
2020-05-20 11:25:42 +00:00
|
|
|
from .const import CONF_KEY, CONF_UUID, KEY_MAC, TIMEOUT
|
2018-12-16 15:19:18 +00:00
|
|
|
|
2018-01-04 10:05:27 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
DOMAIN = "daikin"
|
2018-01-04 10:05:27 +00:00
|
|
|
|
2019-03-27 20:37:21 +00:00
|
|
|
PARALLEL_UPDATES = 0
|
2018-01-04 10:05:27 +00:00
|
|
|
MIN_TIME_BETWEEN_UPDATES = timedelta(seconds=60)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
COMPONENT_TYPES = ["climate", "sensor", "switch"]
|
2018-01-04 10:05:27 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
CONFIG_SCHEMA = vol.Schema(
|
2020-05-23 09:08:49 +00:00
|
|
|
vol.All(
|
|
|
|
cv.deprecated(DOMAIN, invalidation_version="0.113.0"),
|
|
|
|
{
|
|
|
|
DOMAIN: vol.Schema(
|
|
|
|
{
|
|
|
|
vol.Optional(CONF_HOSTS, default=[]): vol.All(
|
|
|
|
cv.ensure_list, [cv.string]
|
|
|
|
)
|
|
|
|
}
|
|
|
|
)
|
|
|
|
},
|
|
|
|
),
|
2019-07-31 19:25:30 +00:00
|
|
|
extra=vol.ALLOW_EXTRA,
|
|
|
|
)
|
2018-01-04 10:05:27 +00:00
|
|
|
|
|
|
|
|
2018-12-16 15:19:18 +00:00
|
|
|
async def async_setup(hass, config):
|
2018-01-04 10:05:27 +00:00
|
|
|
"""Establish connection with Daikin."""
|
2018-12-16 15:19:18 +00:00
|
|
|
if DOMAIN not in config:
|
|
|
|
return True
|
|
|
|
|
2020-04-15 12:10:07 +00:00
|
|
|
hosts = config[DOMAIN][CONF_HOSTS]
|
2018-12-16 15:19:18 +00:00
|
|
|
if not hosts:
|
|
|
|
hass.async_create_task(
|
|
|
|
hass.config_entries.flow.async_init(
|
2019-07-31 19:25:30 +00:00
|
|
|
DOMAIN, context={"source": SOURCE_IMPORT}
|
|
|
|
)
|
|
|
|
)
|
2018-12-16 15:19:18 +00:00
|
|
|
for host in hosts:
|
|
|
|
hass.async_create_task(
|
|
|
|
hass.config_entries.flow.async_init(
|
2019-07-31 19:25:30 +00:00
|
|
|
DOMAIN, context={"source": SOURCE_IMPORT}, data={CONF_HOST: host}
|
|
|
|
)
|
|
|
|
)
|
2018-12-16 15:19:18 +00:00
|
|
|
return True
|
2018-01-04 10:05:27 +00:00
|
|
|
|
|
|
|
|
2018-12-16 15:19:18 +00:00
|
|
|
async def async_setup_entry(hass: HomeAssistantType, entry: ConfigEntry):
|
|
|
|
"""Establish connection with Daikin."""
|
|
|
|
conf = entry.data
|
2020-05-20 11:25:42 +00:00
|
|
|
# For backwards compat, set unique ID
|
|
|
|
if entry.unique_id is None:
|
|
|
|
hass.config_entries.async_update_entry(entry, unique_id=conf[KEY_MAC])
|
2020-05-06 11:59:07 +00:00
|
|
|
daikin_api = await daikin_api_setup(
|
|
|
|
hass,
|
|
|
|
conf[CONF_HOST],
|
|
|
|
conf.get(CONF_KEY),
|
|
|
|
conf.get(CONF_UUID),
|
|
|
|
conf.get(CONF_PASSWORD),
|
|
|
|
)
|
2018-12-16 15:19:18 +00:00
|
|
|
if not daikin_api:
|
|
|
|
return False
|
|
|
|
hass.data.setdefault(DOMAIN, {}).update({entry.entry_id: daikin_api})
|
2019-04-26 18:24:02 +00:00
|
|
|
for component in COMPONENT_TYPES:
|
|
|
|
hass.async_create_task(
|
2019-07-31 19:25:30 +00:00
|
|
|
hass.config_entries.async_forward_entry_setup(entry, component)
|
|
|
|
)
|
2018-12-16 15:19:18 +00:00
|
|
|
return True
|
2018-01-04 10:05:27 +00:00
|
|
|
|
|
|
|
|
2018-12-16 15:19:18 +00:00
|
|
|
async def async_unload_entry(hass, config_entry):
|
|
|
|
"""Unload a config entry."""
|
2019-07-31 19:25:30 +00:00
|
|
|
await asyncio.wait(
|
|
|
|
[
|
|
|
|
hass.config_entries.async_forward_entry_unload(config_entry, component)
|
|
|
|
for component in COMPONENT_TYPES
|
|
|
|
]
|
|
|
|
)
|
2018-12-16 15:19:18 +00:00
|
|
|
hass.data[DOMAIN].pop(config_entry.entry_id)
|
|
|
|
if not hass.data[DOMAIN]:
|
|
|
|
hass.data.pop(DOMAIN)
|
2018-01-04 10:05:27 +00:00
|
|
|
return True
|
|
|
|
|
|
|
|
|
2020-05-06 11:59:07 +00:00
|
|
|
async def daikin_api_setup(hass, host, key, uuid, password):
|
2018-01-04 10:05:27 +00:00
|
|
|
"""Create a Daikin instance only once."""
|
2019-07-31 19:25:30 +00:00
|
|
|
|
2019-03-14 17:33:43 +00:00
|
|
|
session = hass.helpers.aiohttp_client.async_get_clientsession()
|
2018-12-16 15:19:18 +00:00
|
|
|
try:
|
2020-04-15 13:12:10 +00:00
|
|
|
with timeout(TIMEOUT):
|
2020-05-06 11:59:07 +00:00
|
|
|
device = await Appliance.factory(
|
|
|
|
host, session, key=key, uuid=uuid, password=password
|
|
|
|
)
|
2018-12-16 15:19:18 +00:00
|
|
|
except asyncio.TimeoutError:
|
2019-04-10 07:44:00 +00:00
|
|
|
_LOGGER.debug("Connection to %s timed out", host)
|
|
|
|
raise ConfigEntryNotReady
|
2019-04-08 17:08:03 +00:00
|
|
|
except ClientConnectionError:
|
2019-04-10 07:44:00 +00:00
|
|
|
_LOGGER.debug("ClientConnectionError to %s", host)
|
|
|
|
raise ConfigEntryNotReady
|
2018-12-16 15:19:18 +00:00
|
|
|
except Exception: # pylint: disable=broad-except
|
2019-04-10 07:44:00 +00:00
|
|
|
_LOGGER.error("Unexpected error creating device %s", host)
|
2018-12-16 15:19:18 +00:00
|
|
|
return None
|
|
|
|
|
2019-03-14 17:33:43 +00:00
|
|
|
api = DaikinApi(device)
|
2018-01-04 10:05:27 +00:00
|
|
|
|
|
|
|
return api
|
|
|
|
|
|
|
|
|
2018-07-20 08:45:20 +00:00
|
|
|
class DaikinApi:
|
2018-01-04 10:05:27 +00:00
|
|
|
"""Keep the Daikin instance in one place and centralize the update."""
|
|
|
|
|
2020-05-08 17:21:24 +00:00
|
|
|
def __init__(self, device: Appliance):
|
2018-01-04 10:05:27 +00:00
|
|
|
"""Initialize the Daikin Handle."""
|
|
|
|
self.device = device
|
2020-05-06 11:59:07 +00:00
|
|
|
self.name = device.values.get("name", "Daikin AC")
|
|
|
|
self.ip_address = device.device_ip
|
2019-03-14 17:33:43 +00:00
|
|
|
self._available = True
|
2018-01-04 10:05:27 +00:00
|
|
|
|
|
|
|
@Throttle(MIN_TIME_BETWEEN_UPDATES)
|
2019-03-14 17:33:43 +00:00
|
|
|
async def async_update(self, **kwargs):
|
2018-01-04 10:05:27 +00:00
|
|
|
"""Pull the latest data from Daikin."""
|
|
|
|
try:
|
2019-03-14 17:33:43 +00:00
|
|
|
await self.device.update_status()
|
|
|
|
self._available = True
|
2019-04-08 17:08:03 +00:00
|
|
|
except ClientConnectionError:
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.warning("Connection failed for %s", self.ip_address)
|
2019-03-14 17:33:43 +00:00
|
|
|
self._available = False
|
|
|
|
|
|
|
|
@property
|
|
|
|
def available(self) -> bool:
|
|
|
|
"""Return True if entity is available."""
|
|
|
|
return self._available
|
2018-11-27 14:36:55 +00:00
|
|
|
|
2018-12-19 07:18:40 +00:00
|
|
|
@property
|
|
|
|
def device_info(self):
|
|
|
|
"""Return a device description for device registry."""
|
|
|
|
info = self.device.values
|
|
|
|
return {
|
2020-05-06 11:59:07 +00:00
|
|
|
"identifieres": self.device.mac,
|
2019-07-31 19:25:30 +00:00
|
|
|
"manufacturer": "Daikin",
|
|
|
|
"model": info.get("model"),
|
|
|
|
"name": info.get("name"),
|
2020-05-06 11:59:07 +00:00
|
|
|
"sw_version": info.get("ver", "").replace("_", "."),
|
2018-12-19 07:18:40 +00:00
|
|
|
}
|