2019-02-14 04:35:12 +00:00
|
|
|
"""Support for Nest devices."""
|
2020-10-21 08:17:49 +00:00
|
|
|
|
|
|
|
import asyncio
|
2016-04-12 04:52:19 +00:00
|
|
|
import logging
|
|
|
|
|
2020-11-30 08:19:42 +00:00
|
|
|
from google_nest_sdm.event import AsyncEventCallback, EventMessage
|
2020-12-20 00:41:29 +00:00
|
|
|
from google_nest_sdm.exceptions import AuthException, GoogleNestException
|
2020-10-21 08:17:49 +00:00
|
|
|
from google_nest_sdm.google_nest_subscriber import GoogleNestSubscriber
|
2016-04-01 14:31:11 +00:00
|
|
|
import voluptuous as vol
|
2016-09-01 20:08:03 +00:00
|
|
|
|
2020-12-20 00:41:29 +00:00
|
|
|
from homeassistant.config_entries import SOURCE_REAUTH, ConfigEntry
|
2017-04-30 05:04:49 +00:00
|
|
|
from homeassistant.const import (
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_BINARY_SENSORS,
|
2020-05-30 15:27:20 +00:00
|
|
|
CONF_CLIENT_ID,
|
|
|
|
CONF_CLIENT_SECRET,
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_MONITORED_CONDITIONS,
|
|
|
|
CONF_SENSORS,
|
|
|
|
CONF_STRUCTURE,
|
|
|
|
)
|
2020-12-22 20:42:37 +00:00
|
|
|
from homeassistant.core import HomeAssistant
|
2020-11-19 11:26:49 +00:00
|
|
|
from homeassistant.exceptions import ConfigEntryNotReady
|
2020-10-21 08:17:49 +00:00
|
|
|
from homeassistant.helpers import (
|
|
|
|
aiohttp_client,
|
|
|
|
config_entry_oauth2_flow,
|
|
|
|
config_validation as cv,
|
|
|
|
)
|
2020-12-22 20:42:37 +00:00
|
|
|
from homeassistant.helpers.dispatcher import async_dispatcher_send
|
2016-01-14 04:05:47 +00:00
|
|
|
|
2020-12-22 20:42:37 +00:00
|
|
|
from . import api, config_flow
|
2020-10-21 08:17:49 +00:00
|
|
|
from .const import (
|
|
|
|
API_URL,
|
|
|
|
DATA_SDM,
|
2020-11-30 08:19:42 +00:00
|
|
|
DATA_SUBSCRIBER,
|
2020-10-21 08:17:49 +00:00
|
|
|
DOMAIN,
|
|
|
|
OAUTH2_AUTHORIZE,
|
|
|
|
OAUTH2_TOKEN,
|
|
|
|
SIGNAL_NEST_UPDATE,
|
|
|
|
)
|
2020-11-30 08:19:42 +00:00
|
|
|
from .events import EVENT_NAME_MAP, NEST_EVENT
|
2020-12-22 20:42:37 +00:00
|
|
|
from .legacy import async_setup_legacy, async_setup_legacy_entry
|
2018-06-13 15:14:52 +00:00
|
|
|
|
2016-11-28 00:18:47 +00:00
|
|
|
_CONFIGURING = {}
|
2016-09-01 20:08:03 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
2016-01-14 04:05:47 +00:00
|
|
|
|
2020-10-21 08:17:49 +00:00
|
|
|
CONF_PROJECT_ID = "project_id"
|
|
|
|
CONF_SUBSCRIBER_ID = "subscriber_id"
|
2019-07-31 19:25:30 +00:00
|
|
|
DATA_NEST_CONFIG = "nest_config"
|
|
|
|
|
|
|
|
SENSOR_SCHEMA = vol.Schema(
|
|
|
|
{vol.Optional(CONF_MONITORED_CONDITIONS): vol.All(cv.ensure_list)}
|
|
|
|
)
|
|
|
|
|
|
|
|
CONFIG_SCHEMA = vol.Schema(
|
|
|
|
{
|
|
|
|
DOMAIN: vol.Schema(
|
|
|
|
{
|
|
|
|
vol.Required(CONF_CLIENT_ID): cv.string,
|
|
|
|
vol.Required(CONF_CLIENT_SECRET): cv.string,
|
2020-10-21 08:17:49 +00:00
|
|
|
# Required to use the new API (optional for compatibility)
|
|
|
|
vol.Optional(CONF_PROJECT_ID): cv.string,
|
|
|
|
vol.Optional(CONF_SUBSCRIBER_ID): cv.string,
|
|
|
|
# Config that only currently works on the old API
|
2019-07-31 19:25:30 +00:00
|
|
|
vol.Optional(CONF_STRUCTURE): vol.All(cv.ensure_list, [cv.string]),
|
|
|
|
vol.Optional(CONF_SENSORS): SENSOR_SCHEMA,
|
|
|
|
vol.Optional(CONF_BINARY_SENSORS): SENSOR_SCHEMA,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
},
|
|
|
|
extra=vol.ALLOW_EXTRA,
|
|
|
|
)
|
|
|
|
|
2020-11-06 08:57:02 +00:00
|
|
|
# Platforms for SDM API
|
|
|
|
PLATFORMS = ["sensor", "camera", "climate"]
|
2020-10-21 08:17:49 +00:00
|
|
|
|
2016-04-12 04:52:19 +00:00
|
|
|
|
2020-10-21 08:17:49 +00:00
|
|
|
async def async_setup(hass: HomeAssistant, config: dict):
|
|
|
|
"""Set up Nest components with dispatch between old/new flows."""
|
|
|
|
hass.data[DOMAIN] = {}
|
|
|
|
|
|
|
|
if DOMAIN not in config:
|
|
|
|
return True
|
|
|
|
|
|
|
|
if CONF_PROJECT_ID not in config[DOMAIN]:
|
|
|
|
return await async_setup_legacy(hass, config)
|
|
|
|
|
|
|
|
if CONF_SUBSCRIBER_ID not in config[DOMAIN]:
|
|
|
|
_LOGGER.error("Configuration option '{CONF_SUBSCRIBER_ID}' required")
|
|
|
|
return False
|
|
|
|
|
|
|
|
# For setup of ConfigEntry below
|
|
|
|
hass.data[DOMAIN][DATA_NEST_CONFIG] = config[DOMAIN]
|
|
|
|
project_id = config[DOMAIN][CONF_PROJECT_ID]
|
|
|
|
config_flow.NestFlowHandler.register_sdm_api(hass)
|
|
|
|
config_flow.NestFlowHandler.async_register_implementation(
|
|
|
|
hass,
|
|
|
|
config_entry_oauth2_flow.LocalOAuth2Implementation(
|
|
|
|
hass,
|
|
|
|
DOMAIN,
|
|
|
|
config[DOMAIN][CONF_CLIENT_ID],
|
|
|
|
config[DOMAIN][CONF_CLIENT_SECRET],
|
|
|
|
OAUTH2_AUTHORIZE.format(project_id=project_id),
|
|
|
|
OAUTH2_TOKEN,
|
|
|
|
),
|
|
|
|
)
|
|
|
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
2020-11-24 15:53:50 +00:00
|
|
|
class SignalUpdateCallback(AsyncEventCallback):
|
2020-10-21 08:17:49 +00:00
|
|
|
"""An EventCallback invoked when new events arrive from subscriber."""
|
|
|
|
|
|
|
|
def __init__(self, hass: HomeAssistant):
|
|
|
|
"""Initialize EventCallback."""
|
|
|
|
self._hass = hass
|
|
|
|
|
2020-11-24 15:53:50 +00:00
|
|
|
async def async_handle_event(self, event_message: EventMessage):
|
2020-10-21 08:17:49 +00:00
|
|
|
"""Process an incoming EventMessage."""
|
2020-11-24 22:34:43 +00:00
|
|
|
if not event_message.resource_update_name:
|
|
|
|
_LOGGER.debug("Ignoring event with no device_id")
|
|
|
|
return
|
|
|
|
device_id = event_message.resource_update_name
|
|
|
|
_LOGGER.debug("Update for %s @ %s", device_id, event_message.timestamp)
|
2020-10-21 08:17:49 +00:00
|
|
|
traits = event_message.resource_update_traits
|
|
|
|
if traits:
|
|
|
|
_LOGGER.debug("Trait update %s", traits.keys())
|
2020-11-24 22:34:43 +00:00
|
|
|
# This event triggered an update to a device that changed some
|
|
|
|
# properties which the DeviceManager should already have received.
|
|
|
|
# Send a signal to refresh state of all listening devices.
|
|
|
|
async_dispatcher_send(self._hass, SIGNAL_NEST_UPDATE)
|
2020-10-21 08:17:49 +00:00
|
|
|
events = event_message.resource_update_events
|
2020-11-24 22:34:43 +00:00
|
|
|
if not events:
|
|
|
|
return
|
|
|
|
_LOGGER.debug("Event Update %s", events.keys())
|
|
|
|
device_registry = await self._hass.helpers.device_registry.async_get_registry()
|
|
|
|
device_entry = device_registry.async_get_device({(DOMAIN, device_id)}, ())
|
|
|
|
if not device_entry:
|
|
|
|
_LOGGER.debug("Ignoring event for unregistered device '%s'", device_id)
|
2020-10-21 08:17:49 +00:00
|
|
|
return
|
2020-11-24 22:34:43 +00:00
|
|
|
for event in events:
|
2020-11-30 08:19:42 +00:00
|
|
|
event_type = EVENT_NAME_MAP.get(event)
|
|
|
|
if not event_type:
|
2020-11-24 22:34:43 +00:00
|
|
|
continue
|
|
|
|
message = {
|
|
|
|
"device_id": device_entry.id,
|
2020-11-30 08:19:42 +00:00
|
|
|
"type": event_type,
|
2020-11-24 22:34:43 +00:00
|
|
|
}
|
|
|
|
self._hass.bus.async_fire(NEST_EVENT, message)
|
2020-10-21 08:17:49 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry):
|
|
|
|
"""Set up Nest from a config entry with dispatch between old/new flows."""
|
|
|
|
|
|
|
|
if DATA_SDM not in entry.data:
|
|
|
|
return await async_setup_legacy_entry(hass, entry)
|
|
|
|
|
|
|
|
implementation = (
|
|
|
|
await config_entry_oauth2_flow.async_get_config_entry_implementation(
|
|
|
|
hass, entry
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
|
|
|
config = hass.data[DOMAIN][DATA_NEST_CONFIG]
|
|
|
|
|
|
|
|
session = config_entry_oauth2_flow.OAuth2Session(hass, entry, implementation)
|
|
|
|
auth = api.AsyncConfigEntryAuth(
|
|
|
|
aiohttp_client.async_get_clientsession(hass),
|
|
|
|
session,
|
|
|
|
API_URL,
|
|
|
|
)
|
|
|
|
subscriber = GoogleNestSubscriber(
|
|
|
|
auth, config[CONF_PROJECT_ID], config[CONF_SUBSCRIBER_ID]
|
|
|
|
)
|
|
|
|
subscriber.set_update_callback(SignalUpdateCallback(hass))
|
2020-11-19 11:26:49 +00:00
|
|
|
|
|
|
|
try:
|
|
|
|
await subscriber.start_async()
|
2020-12-20 00:41:29 +00:00
|
|
|
except AuthException as err:
|
|
|
|
_LOGGER.debug("Subscriber authentication error: %s", err)
|
|
|
|
hass.async_create_task(
|
|
|
|
hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN,
|
|
|
|
context={"source": SOURCE_REAUTH},
|
|
|
|
data=entry.data,
|
|
|
|
)
|
|
|
|
)
|
|
|
|
return False
|
2020-11-19 11:26:49 +00:00
|
|
|
except GoogleNestException as err:
|
|
|
|
_LOGGER.error("Subscriber error: %s", err)
|
|
|
|
subscriber.stop_async()
|
|
|
|
raise ConfigEntryNotReady from err
|
|
|
|
|
|
|
|
try:
|
|
|
|
await subscriber.async_get_device_manager()
|
|
|
|
except GoogleNestException as err:
|
|
|
|
_LOGGER.error("Device Manager error: %s", err)
|
|
|
|
subscriber.stop_async()
|
|
|
|
raise ConfigEntryNotReady from err
|
|
|
|
|
2020-11-30 08:19:42 +00:00
|
|
|
hass.data[DOMAIN][DATA_SUBSCRIBER] = subscriber
|
2020-10-21 08:17:49 +00:00
|
|
|
|
|
|
|
for component in PLATFORMS:
|
|
|
|
hass.async_create_task(
|
|
|
|
hass.config_entries.async_forward_entry_setup(entry, component)
|
|
|
|
)
|
|
|
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
|
|
async def async_unload_entry(hass: HomeAssistant, entry: ConfigEntry):
|
|
|
|
"""Unload a config entry."""
|
|
|
|
if DATA_SDM not in entry.data:
|
|
|
|
# Legacy API
|
|
|
|
return True
|
|
|
|
|
2020-11-30 08:19:42 +00:00
|
|
|
subscriber = hass.data[DOMAIN][DATA_SUBSCRIBER]
|
2020-10-21 08:17:49 +00:00
|
|
|
subscriber.stop_async()
|
|
|
|
unload_ok = all(
|
|
|
|
await asyncio.gather(
|
|
|
|
*[
|
|
|
|
hass.config_entries.async_forward_entry_unload(entry, component)
|
|
|
|
for component in PLATFORMS
|
|
|
|
]
|
|
|
|
)
|
|
|
|
)
|
|
|
|
if unload_ok:
|
2020-11-30 08:19:42 +00:00
|
|
|
hass.data[DOMAIN].pop(DATA_SUBSCRIBER)
|
2020-10-21 08:17:49 +00:00
|
|
|
|
|
|
|
return unload_ok
|