2017-08-29 20:40:08 +00:00
|
|
|
"""The HTTP api to control the cloud integration."""
|
|
|
|
import asyncio
|
2017-09-12 16:47:04 +00:00
|
|
|
from functools import wraps
|
2017-08-29 20:40:08 +00:00
|
|
|
import logging
|
|
|
|
|
2019-03-12 14:54:04 +00:00
|
|
|
import attr
|
2018-11-26 13:10:18 +00:00
|
|
|
import aiohttp
|
2017-08-29 20:40:08 +00:00
|
|
|
import async_timeout
|
2018-01-21 06:35:38 +00:00
|
|
|
import voluptuous as vol
|
2017-08-29 20:40:08 +00:00
|
|
|
|
2018-09-20 12:53:13 +00:00
|
|
|
from homeassistant.core import callback
|
2018-02-14 20:06:03 +00:00
|
|
|
from homeassistant.components.http import HomeAssistantView
|
|
|
|
from homeassistant.components.http.data_validator import (
|
|
|
|
RequestDataValidator)
|
2018-09-20 12:53:13 +00:00
|
|
|
from homeassistant.components import websocket_api
|
2019-06-13 15:43:57 +00:00
|
|
|
from homeassistant.components.alexa import entities as alexa_entities
|
2019-05-29 15:39:12 +00:00
|
|
|
from homeassistant.components.google_assistant import helpers as google_helpers
|
2017-08-29 20:40:08 +00:00
|
|
|
|
2018-11-20 22:23:07 +00:00
|
|
|
from .const import (
|
|
|
|
DOMAIN, REQUEST_TIMEOUT, PREF_ENABLE_ALEXA, PREF_ENABLE_GOOGLE,
|
2019-06-08 06:08:55 +00:00
|
|
|
PREF_GOOGLE_SECURE_DEVICES_PIN, InvalidTrustedNetworks,
|
2019-06-17 20:50:01 +00:00
|
|
|
InvalidTrustedProxies, PREF_ALEXA_REPORT_STATE)
|
2017-08-29 20:40:08 +00:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
2018-09-20 12:53:13 +00:00
|
|
|
WS_TYPE_STATUS = 'cloud/status'
|
|
|
|
SCHEMA_WS_STATUS = websocket_api.BASE_COMMAND_MESSAGE_SCHEMA.extend({
|
|
|
|
vol.Required('type'): WS_TYPE_STATUS,
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
WS_TYPE_SUBSCRIPTION = 'cloud/subscription'
|
|
|
|
SCHEMA_WS_SUBSCRIPTION = websocket_api.BASE_COMMAND_MESSAGE_SCHEMA.extend({
|
|
|
|
vol.Required('type'): WS_TYPE_SUBSCRIPTION,
|
|
|
|
})
|
|
|
|
|
|
|
|
|
2018-11-26 13:10:18 +00:00
|
|
|
WS_TYPE_HOOK_CREATE = 'cloud/cloudhook/create'
|
|
|
|
SCHEMA_WS_HOOK_CREATE = websocket_api.BASE_COMMAND_MESSAGE_SCHEMA.extend({
|
|
|
|
vol.Required('type'): WS_TYPE_HOOK_CREATE,
|
|
|
|
vol.Required('webhook_id'): str
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
WS_TYPE_HOOK_DELETE = 'cloud/cloudhook/delete'
|
|
|
|
SCHEMA_WS_HOOK_DELETE = websocket_api.BASE_COMMAND_MESSAGE_SCHEMA.extend({
|
|
|
|
vol.Required('type'): WS_TYPE_HOOK_DELETE,
|
|
|
|
vol.Required('webhook_id'): str
|
|
|
|
})
|
|
|
|
|
|
|
|
|
2019-03-16 02:26:10 +00:00
|
|
|
_CLOUD_ERRORS = {
|
|
|
|
InvalidTrustedNetworks:
|
|
|
|
(500, 'Remote UI not compatible with 127.0.0.1/::1'
|
2019-06-08 06:08:55 +00:00
|
|
|
' as a trusted network.'),
|
|
|
|
InvalidTrustedProxies:
|
|
|
|
(500, 'Remote UI not compatible with 127.0.0.1/::1'
|
|
|
|
' as trusted proxies.'),
|
2019-03-16 02:26:10 +00:00
|
|
|
}
|
2019-03-11 19:21:20 +00:00
|
|
|
|
|
|
|
|
2018-03-23 19:13:52 +00:00
|
|
|
async def async_setup(hass):
|
2018-01-21 06:35:38 +00:00
|
|
|
"""Initialize the HTTP API."""
|
2018-09-20 12:53:13 +00:00
|
|
|
hass.components.websocket_api.async_register_command(
|
|
|
|
WS_TYPE_STATUS, websocket_cloud_status,
|
|
|
|
SCHEMA_WS_STATUS
|
|
|
|
)
|
|
|
|
hass.components.websocket_api.async_register_command(
|
|
|
|
WS_TYPE_SUBSCRIPTION, websocket_subscription,
|
|
|
|
SCHEMA_WS_SUBSCRIPTION
|
|
|
|
)
|
2018-09-20 21:46:51 +00:00
|
|
|
hass.components.websocket_api.async_register_command(
|
2019-04-19 21:50:21 +00:00
|
|
|
websocket_update_prefs)
|
2018-11-26 13:10:18 +00:00
|
|
|
hass.components.websocket_api.async_register_command(
|
|
|
|
WS_TYPE_HOOK_CREATE, websocket_hook_create,
|
|
|
|
SCHEMA_WS_HOOK_CREATE
|
|
|
|
)
|
|
|
|
hass.components.websocket_api.async_register_command(
|
|
|
|
WS_TYPE_HOOK_DELETE, websocket_hook_delete,
|
|
|
|
SCHEMA_WS_HOOK_DELETE
|
|
|
|
)
|
2019-03-12 14:54:04 +00:00
|
|
|
hass.components.websocket_api.async_register_command(
|
|
|
|
websocket_remote_connect)
|
|
|
|
hass.components.websocket_api.async_register_command(
|
|
|
|
websocket_remote_disconnect)
|
2019-05-29 15:39:12 +00:00
|
|
|
|
|
|
|
hass.components.websocket_api.async_register_command(
|
|
|
|
google_assistant_list)
|
|
|
|
hass.components.websocket_api.async_register_command(
|
|
|
|
google_assistant_update)
|
|
|
|
|
2019-06-13 18:58:08 +00:00
|
|
|
hass.components.websocket_api.async_register_command(alexa_list)
|
|
|
|
hass.components.websocket_api.async_register_command(alexa_update)
|
|
|
|
|
2018-03-23 19:13:52 +00:00
|
|
|
hass.http.register_view(GoogleActionsSyncView)
|
2017-08-29 20:40:08 +00:00
|
|
|
hass.http.register_view(CloudLoginView)
|
|
|
|
hass.http.register_view(CloudLogoutView)
|
2017-09-12 16:47:04 +00:00
|
|
|
hass.http.register_view(CloudRegisterView)
|
2017-12-29 13:46:10 +00:00
|
|
|
hass.http.register_view(CloudResendConfirmView)
|
2017-09-12 16:47:04 +00:00
|
|
|
hass.http.register_view(CloudForgotPasswordView)
|
2017-08-29 20:40:08 +00:00
|
|
|
|
2019-03-11 19:21:20 +00:00
|
|
|
from hass_nabucasa import auth
|
2017-08-29 20:40:08 +00:00
|
|
|
|
2019-03-11 19:21:20 +00:00
|
|
|
_CLOUD_ERRORS.update({
|
|
|
|
auth.UserNotFound:
|
|
|
|
(400, "User does not exist."),
|
|
|
|
auth.UserNotConfirmed:
|
|
|
|
(400, 'Email not confirmed.'),
|
2019-04-17 13:57:26 +00:00
|
|
|
auth.UserExists:
|
|
|
|
(400, 'An account with the given email already exists.'),
|
2019-03-11 19:21:20 +00:00
|
|
|
auth.Unauthenticated:
|
|
|
|
(401, 'Authentication failed.'),
|
|
|
|
auth.PasswordChangeRequired:
|
|
|
|
(400, 'Password change required.'),
|
|
|
|
asyncio.TimeoutError:
|
2019-03-16 02:26:10 +00:00
|
|
|
(502, 'Unable to reach the Home Assistant cloud.'),
|
|
|
|
aiohttp.ClientError:
|
|
|
|
(500, 'Error making internal request'),
|
2019-03-11 19:21:20 +00:00
|
|
|
})
|
2017-08-29 20:40:08 +00:00
|
|
|
|
|
|
|
|
2017-09-12 16:47:04 +00:00
|
|
|
def _handle_cloud_errors(handler):
|
2018-11-26 13:10:18 +00:00
|
|
|
"""Webview decorator to handle auth errors."""
|
2017-09-12 16:47:04 +00:00
|
|
|
@wraps(handler)
|
2018-03-23 19:13:52 +00:00
|
|
|
async def error_handler(view, request, *args, **kwargs):
|
2017-09-12 16:47:04 +00:00
|
|
|
"""Handle exceptions that raise from the wrapped request handler."""
|
2017-08-29 20:40:08 +00:00
|
|
|
try:
|
2018-03-23 19:13:52 +00:00
|
|
|
result = await handler(view, request, *args, **kwargs)
|
2017-09-12 16:47:04 +00:00
|
|
|
return result
|
2017-08-29 20:40:08 +00:00
|
|
|
|
2019-02-04 09:14:30 +00:00
|
|
|
except Exception as err: # pylint: disable=broad-except
|
2019-03-16 02:26:10 +00:00
|
|
|
status, msg = _process_cloud_exception(err, request.path)
|
2019-02-04 09:14:30 +00:00
|
|
|
return view.json_message(
|
|
|
|
msg, status_code=status,
|
|
|
|
message_code=err.__class__.__name__.lower())
|
2017-08-29 20:40:08 +00:00
|
|
|
|
2017-09-12 16:47:04 +00:00
|
|
|
return error_handler
|
2017-08-29 20:40:08 +00:00
|
|
|
|
|
|
|
|
2019-03-16 02:26:10 +00:00
|
|
|
def _ws_handle_cloud_errors(handler):
|
|
|
|
"""Websocket decorator to handle auth errors."""
|
|
|
|
@wraps(handler)
|
|
|
|
async def error_handler(hass, connection, msg):
|
|
|
|
"""Handle exceptions that raise from the wrapped handler."""
|
|
|
|
try:
|
|
|
|
return await handler(hass, connection, msg)
|
|
|
|
|
|
|
|
except Exception as err: # pylint: disable=broad-except
|
|
|
|
err_status, err_msg = _process_cloud_exception(err, msg['type'])
|
|
|
|
connection.send_error(msg['id'], err_status, err_msg)
|
|
|
|
|
|
|
|
return error_handler
|
|
|
|
|
|
|
|
|
|
|
|
def _process_cloud_exception(exc, where):
|
|
|
|
"""Process a cloud exception."""
|
|
|
|
err_info = _CLOUD_ERRORS.get(exc.__class__)
|
|
|
|
if err_info is None:
|
|
|
|
_LOGGER.exception(
|
|
|
|
"Unexpected error processing request for %s", where)
|
|
|
|
err_info = (502, 'Unexpected error: {}'.format(exc))
|
|
|
|
return err_info
|
|
|
|
|
|
|
|
|
2018-03-23 19:13:52 +00:00
|
|
|
class GoogleActionsSyncView(HomeAssistantView):
|
|
|
|
"""Trigger a Google Actions Smart Home Sync."""
|
|
|
|
|
|
|
|
url = '/api/cloud/google_actions/sync'
|
|
|
|
name = 'api:cloud:google_actions/sync'
|
|
|
|
|
|
|
|
@_handle_cloud_errors
|
|
|
|
async def post(self, request):
|
|
|
|
"""Trigger a Google Actions sync."""
|
|
|
|
hass = request.app['hass']
|
|
|
|
cloud = hass.data[DOMAIN]
|
|
|
|
websession = hass.helpers.aiohttp_client.async_get_clientsession()
|
|
|
|
|
2019-05-23 04:09:59 +00:00
|
|
|
with async_timeout.timeout(REQUEST_TIMEOUT):
|
2019-03-11 19:21:20 +00:00
|
|
|
await hass.async_add_job(cloud.auth.check_token)
|
2018-03-23 19:13:52 +00:00
|
|
|
|
2019-05-23 04:09:59 +00:00
|
|
|
with async_timeout.timeout(REQUEST_TIMEOUT):
|
2018-03-23 19:13:52 +00:00
|
|
|
req = await websession.post(
|
|
|
|
cloud.google_actions_sync_url, headers={
|
|
|
|
'authorization': cloud.id_token
|
|
|
|
})
|
|
|
|
|
|
|
|
return self.json({}, status_code=req.status)
|
|
|
|
|
|
|
|
|
2017-09-12 16:47:04 +00:00
|
|
|
class CloudLoginView(HomeAssistantView):
|
|
|
|
"""Login to Home Assistant cloud."""
|
2017-08-29 20:40:08 +00:00
|
|
|
|
2017-09-12 16:47:04 +00:00
|
|
|
url = '/api/cloud/login'
|
|
|
|
name = 'api:cloud:login'
|
2017-08-29 20:40:08 +00:00
|
|
|
|
2017-09-12 16:47:04 +00:00
|
|
|
@_handle_cloud_errors
|
|
|
|
@RequestDataValidator(vol.Schema({
|
|
|
|
vol.Required('email'): str,
|
|
|
|
vol.Required('password'): str,
|
|
|
|
}))
|
2018-03-23 19:13:52 +00:00
|
|
|
async def post(self, request, data):
|
2017-09-12 16:47:04 +00:00
|
|
|
"""Handle login request."""
|
|
|
|
hass = request.app['hass']
|
2017-10-15 02:43:14 +00:00
|
|
|
cloud = hass.data[DOMAIN]
|
2017-09-12 16:47:04 +00:00
|
|
|
|
2019-05-23 04:09:59 +00:00
|
|
|
with async_timeout.timeout(REQUEST_TIMEOUT):
|
2019-03-11 19:21:20 +00:00
|
|
|
await hass.async_add_job(cloud.auth.login, data['email'],
|
2018-03-23 19:13:52 +00:00
|
|
|
data['password'])
|
2017-09-12 16:47:04 +00:00
|
|
|
|
2017-11-15 07:16:19 +00:00
|
|
|
hass.async_add_job(cloud.iot.connect)
|
2018-09-20 12:53:13 +00:00
|
|
|
return self.json({'success': True})
|
2017-08-29 20:40:08 +00:00
|
|
|
|
|
|
|
|
|
|
|
class CloudLogoutView(HomeAssistantView):
|
|
|
|
"""Log out of the Home Assistant cloud."""
|
|
|
|
|
|
|
|
url = '/api/cloud/logout'
|
|
|
|
name = 'api:cloud:logout'
|
|
|
|
|
2017-09-12 16:47:04 +00:00
|
|
|
@_handle_cloud_errors
|
2018-03-23 19:13:52 +00:00
|
|
|
async def post(self, request):
|
2017-09-12 16:47:04 +00:00
|
|
|
"""Handle logout request."""
|
2017-08-29 20:40:08 +00:00
|
|
|
hass = request.app['hass']
|
2017-10-15 02:43:14 +00:00
|
|
|
cloud = hass.data[DOMAIN]
|
2017-08-29 20:40:08 +00:00
|
|
|
|
2019-05-23 04:09:59 +00:00
|
|
|
with async_timeout.timeout(REQUEST_TIMEOUT):
|
2018-03-23 19:13:52 +00:00
|
|
|
await cloud.logout()
|
2017-08-29 20:40:08 +00:00
|
|
|
|
2017-09-12 16:47:04 +00:00
|
|
|
return self.json_message('ok')
|
2017-08-29 20:40:08 +00:00
|
|
|
|
|
|
|
|
2017-09-12 16:47:04 +00:00
|
|
|
class CloudRegisterView(HomeAssistantView):
|
|
|
|
"""Register on the Home Assistant cloud."""
|
|
|
|
|
|
|
|
url = '/api/cloud/register'
|
|
|
|
name = 'api:cloud:register'
|
|
|
|
|
|
|
|
@_handle_cloud_errors
|
|
|
|
@RequestDataValidator(vol.Schema({
|
|
|
|
vol.Required('email'): str,
|
|
|
|
vol.Required('password'): vol.All(str, vol.Length(min=6)),
|
|
|
|
}))
|
2018-03-23 19:13:52 +00:00
|
|
|
async def post(self, request, data):
|
2017-09-12 16:47:04 +00:00
|
|
|
"""Handle registration request."""
|
|
|
|
hass = request.app['hass']
|
2017-10-15 02:43:14 +00:00
|
|
|
cloud = hass.data[DOMAIN]
|
2017-09-12 16:47:04 +00:00
|
|
|
|
2019-05-23 04:09:59 +00:00
|
|
|
with async_timeout.timeout(REQUEST_TIMEOUT):
|
2018-03-23 19:13:52 +00:00
|
|
|
await hass.async_add_job(
|
2019-03-11 19:21:20 +00:00
|
|
|
cloud.auth.register, data['email'], data['password'])
|
2017-09-12 16:47:04 +00:00
|
|
|
|
|
|
|
return self.json_message('ok')
|
|
|
|
|
|
|
|
|
2017-12-29 13:46:10 +00:00
|
|
|
class CloudResendConfirmView(HomeAssistantView):
|
|
|
|
"""Resend email confirmation code."""
|
|
|
|
|
|
|
|
url = '/api/cloud/resend_confirm'
|
|
|
|
name = 'api:cloud:resend_confirm'
|
|
|
|
|
|
|
|
@_handle_cloud_errors
|
|
|
|
@RequestDataValidator(vol.Schema({
|
|
|
|
vol.Required('email'): str,
|
|
|
|
}))
|
2018-03-23 19:13:52 +00:00
|
|
|
async def post(self, request, data):
|
2017-12-29 13:46:10 +00:00
|
|
|
"""Handle resending confirm email code request."""
|
|
|
|
hass = request.app['hass']
|
|
|
|
cloud = hass.data[DOMAIN]
|
|
|
|
|
2019-05-23 04:09:59 +00:00
|
|
|
with async_timeout.timeout(REQUEST_TIMEOUT):
|
2018-03-23 19:13:52 +00:00
|
|
|
await hass.async_add_job(
|
2019-03-11 19:21:20 +00:00
|
|
|
cloud.auth.resend_email_confirm, data['email'])
|
2017-12-29 13:46:10 +00:00
|
|
|
|
|
|
|
return self.json_message('ok')
|
|
|
|
|
|
|
|
|
2017-09-12 16:47:04 +00:00
|
|
|
class CloudForgotPasswordView(HomeAssistantView):
|
|
|
|
"""View to start Forgot Password flow.."""
|
|
|
|
|
|
|
|
url = '/api/cloud/forgot_password'
|
|
|
|
name = 'api:cloud:forgot_password'
|
|
|
|
|
|
|
|
@_handle_cloud_errors
|
|
|
|
@RequestDataValidator(vol.Schema({
|
|
|
|
vol.Required('email'): str,
|
|
|
|
}))
|
2018-03-23 19:13:52 +00:00
|
|
|
async def post(self, request, data):
|
2017-09-12 16:47:04 +00:00
|
|
|
"""Handle forgot password request."""
|
|
|
|
hass = request.app['hass']
|
2017-10-15 02:43:14 +00:00
|
|
|
cloud = hass.data[DOMAIN]
|
2017-09-12 16:47:04 +00:00
|
|
|
|
2019-05-23 04:09:59 +00:00
|
|
|
with async_timeout.timeout(REQUEST_TIMEOUT):
|
2018-03-23 19:13:52 +00:00
|
|
|
await hass.async_add_job(
|
2019-03-11 19:21:20 +00:00
|
|
|
cloud.auth.forgot_password, data['email'])
|
2017-09-12 16:47:04 +00:00
|
|
|
|
|
|
|
return self.json_message('ok')
|
|
|
|
|
|
|
|
|
2018-09-20 12:53:13 +00:00
|
|
|
@callback
|
|
|
|
def websocket_cloud_status(hass, connection, msg):
|
|
|
|
"""Handle request for account info.
|
|
|
|
|
|
|
|
Async friendly.
|
|
|
|
"""
|
|
|
|
cloud = hass.data[DOMAIN]
|
2018-10-01 14:09:31 +00:00
|
|
|
connection.send_message(
|
2018-09-20 12:53:13 +00:00
|
|
|
websocket_api.result_message(msg['id'], _account_data(cloud)))
|
|
|
|
|
|
|
|
|
2018-11-26 13:10:18 +00:00
|
|
|
def _require_cloud_login(handler):
|
|
|
|
"""Websocket decorator that requires cloud to be logged in."""
|
|
|
|
@wraps(handler)
|
|
|
|
def with_cloud_auth(hass, connection, msg):
|
|
|
|
"""Require to be logged into the cloud."""
|
|
|
|
cloud = hass.data[DOMAIN]
|
|
|
|
if not cloud.is_logged_in:
|
|
|
|
connection.send_message(websocket_api.error_message(
|
|
|
|
msg['id'], 'not_logged_in',
|
|
|
|
'You need to be logged in to the cloud.'))
|
|
|
|
return
|
|
|
|
|
|
|
|
handler(hass, connection, msg)
|
|
|
|
|
|
|
|
return with_cloud_auth
|
|
|
|
|
|
|
|
|
|
|
|
@_require_cloud_login
|
2018-09-20 12:53:13 +00:00
|
|
|
@websocket_api.async_response
|
|
|
|
async def websocket_subscription(hass, connection, msg):
|
|
|
|
"""Handle request for account info."""
|
2019-03-11 19:21:20 +00:00
|
|
|
from hass_nabucasa.const import STATE_DISCONNECTED
|
2018-09-20 12:53:13 +00:00
|
|
|
cloud = hass.data[DOMAIN]
|
|
|
|
|
2019-05-23 04:09:59 +00:00
|
|
|
with async_timeout.timeout(REQUEST_TIMEOUT):
|
2018-09-20 12:53:13 +00:00
|
|
|
response = await cloud.fetch_subscription_info()
|
|
|
|
|
2018-10-21 10:16:24 +00:00
|
|
|
if response.status != 200:
|
2018-10-01 14:09:31 +00:00
|
|
|
connection.send_message(websocket_api.error_message(
|
2018-09-20 12:53:13 +00:00
|
|
|
msg['id'], 'request_failed', 'Failed to request subscription'))
|
2018-10-21 10:16:24 +00:00
|
|
|
|
|
|
|
data = await response.json()
|
|
|
|
|
|
|
|
# Check if a user is subscribed but local info is outdated
|
|
|
|
# In that case, let's refresh and reconnect
|
2019-03-11 19:21:20 +00:00
|
|
|
if data.get('provider') and not cloud.is_connected:
|
2018-10-21 10:16:24 +00:00
|
|
|
_LOGGER.debug(
|
|
|
|
"Found disconnected account with valid subscriotion, connecting")
|
2019-03-11 19:21:20 +00:00
|
|
|
await hass.async_add_executor_job(cloud.auth.renew_access_token)
|
2018-10-21 10:16:24 +00:00
|
|
|
|
|
|
|
# Cancel reconnect in progress
|
|
|
|
if cloud.iot.state != STATE_DISCONNECTED:
|
|
|
|
await cloud.iot.disconnect()
|
|
|
|
|
|
|
|
hass.async_create_task(cloud.iot.connect())
|
|
|
|
|
|
|
|
connection.send_message(websocket_api.result_message(msg['id'], data))
|
2018-09-20 12:53:13 +00:00
|
|
|
|
|
|
|
|
2018-11-26 13:10:18 +00:00
|
|
|
@_require_cloud_login
|
2018-09-20 21:46:51 +00:00
|
|
|
@websocket_api.async_response
|
2019-04-19 21:50:21 +00:00
|
|
|
@websocket_api.websocket_command({
|
|
|
|
vol.Required('type'): 'cloud/update_prefs',
|
|
|
|
vol.Optional(PREF_ENABLE_GOOGLE): bool,
|
|
|
|
vol.Optional(PREF_ENABLE_ALEXA): bool,
|
2019-06-17 20:50:01 +00:00
|
|
|
vol.Optional(PREF_ALEXA_REPORT_STATE): bool,
|
2019-04-19 21:50:21 +00:00
|
|
|
vol.Optional(PREF_GOOGLE_SECURE_DEVICES_PIN): vol.Any(None, str),
|
|
|
|
})
|
2018-09-20 21:46:51 +00:00
|
|
|
async def websocket_update_prefs(hass, connection, msg):
|
|
|
|
"""Handle request for account info."""
|
|
|
|
cloud = hass.data[DOMAIN]
|
|
|
|
|
|
|
|
changes = dict(msg)
|
|
|
|
changes.pop('id')
|
|
|
|
changes.pop('type')
|
2019-03-11 19:21:20 +00:00
|
|
|
await cloud.client.prefs.async_update(**changes)
|
2018-09-20 21:46:51 +00:00
|
|
|
|
2018-11-26 13:10:18 +00:00
|
|
|
connection.send_message(websocket_api.result_message(msg['id']))
|
|
|
|
|
|
|
|
|
|
|
|
@_require_cloud_login
|
|
|
|
@websocket_api.async_response
|
2019-03-16 02:26:10 +00:00
|
|
|
@_ws_handle_cloud_errors
|
2018-11-26 13:10:18 +00:00
|
|
|
async def websocket_hook_create(hass, connection, msg):
|
|
|
|
"""Handle request for account info."""
|
|
|
|
cloud = hass.data[DOMAIN]
|
2019-03-12 18:49:46 +00:00
|
|
|
hook = await cloud.cloudhooks.async_create(msg['webhook_id'], False)
|
2018-11-26 13:10:18 +00:00
|
|
|
connection.send_message(websocket_api.result_message(msg['id'], hook))
|
|
|
|
|
|
|
|
|
|
|
|
@_require_cloud_login
|
|
|
|
@websocket_api.async_response
|
2019-03-16 02:26:10 +00:00
|
|
|
@_ws_handle_cloud_errors
|
2018-11-26 13:10:18 +00:00
|
|
|
async def websocket_hook_delete(hass, connection, msg):
|
|
|
|
"""Handle request for account info."""
|
|
|
|
cloud = hass.data[DOMAIN]
|
|
|
|
await cloud.cloudhooks.async_delete(msg['webhook_id'])
|
|
|
|
connection.send_message(websocket_api.result_message(msg['id']))
|
2018-09-20 21:46:51 +00:00
|
|
|
|
|
|
|
|
2017-10-15 02:43:14 +00:00
|
|
|
def _account_data(cloud):
|
2017-09-12 16:47:04 +00:00
|
|
|
"""Generate the auth data JSON response."""
|
2019-03-11 19:21:20 +00:00
|
|
|
from hass_nabucasa.const import STATE_DISCONNECTED
|
|
|
|
|
2018-09-20 12:53:13 +00:00
|
|
|
if not cloud.is_logged_in:
|
|
|
|
return {
|
|
|
|
'logged_in': False,
|
|
|
|
'cloud': STATE_DISCONNECTED,
|
|
|
|
}
|
|
|
|
|
2017-11-15 07:16:19 +00:00
|
|
|
claims = cloud.claims
|
2019-03-11 19:21:20 +00:00
|
|
|
client = cloud.client
|
2019-03-12 14:54:04 +00:00
|
|
|
remote = cloud.remote
|
|
|
|
|
|
|
|
# Load remote certificate
|
|
|
|
if remote.certificate:
|
|
|
|
certificate = attr.asdict(remote.certificate)
|
|
|
|
else:
|
|
|
|
certificate = None
|
2017-11-15 07:16:19 +00:00
|
|
|
|
2017-09-12 16:47:04 +00:00
|
|
|
return {
|
2018-09-20 12:53:13 +00:00
|
|
|
'logged_in': True,
|
2017-11-15 07:16:19 +00:00
|
|
|
'email': claims['email'],
|
|
|
|
'cloud': cloud.iot.state,
|
2019-03-11 19:21:20 +00:00
|
|
|
'prefs': client.prefs.as_dict(),
|
|
|
|
'google_entities': client.google_user_config['filter'].config,
|
2019-06-13 18:58:08 +00:00
|
|
|
'alexa_entities': client.alexa_user_config['filter'].config,
|
2019-03-12 14:54:04 +00:00
|
|
|
'remote_domain': remote.instance_domain,
|
|
|
|
'remote_connected': remote.is_connected,
|
|
|
|
'remote_certificate': certificate,
|
2017-09-12 16:47:04 +00:00
|
|
|
}
|
2019-03-12 14:54:04 +00:00
|
|
|
|
|
|
|
|
2019-04-01 08:22:51 +00:00
|
|
|
@websocket_api.require_admin
|
2019-03-12 14:54:04 +00:00
|
|
|
@_require_cloud_login
|
|
|
|
@websocket_api.async_response
|
2019-03-16 02:26:10 +00:00
|
|
|
@_ws_handle_cloud_errors
|
2019-03-12 14:54:04 +00:00
|
|
|
@websocket_api.websocket_command({
|
|
|
|
'type': 'cloud/remote/connect'
|
|
|
|
})
|
|
|
|
async def websocket_remote_connect(hass, connection, msg):
|
|
|
|
"""Handle request for connect remote."""
|
|
|
|
cloud = hass.data[DOMAIN]
|
|
|
|
await cloud.client.prefs.async_update(remote_enabled=True)
|
2019-03-16 02:26:10 +00:00
|
|
|
await cloud.remote.connect()
|
2019-03-12 14:54:04 +00:00
|
|
|
connection.send_result(msg['id'], _account_data(cloud))
|
|
|
|
|
|
|
|
|
2019-04-01 08:22:51 +00:00
|
|
|
@websocket_api.require_admin
|
2019-03-12 14:54:04 +00:00
|
|
|
@_require_cloud_login
|
|
|
|
@websocket_api.async_response
|
2019-03-16 02:26:10 +00:00
|
|
|
@_ws_handle_cloud_errors
|
2019-03-12 14:54:04 +00:00
|
|
|
@websocket_api.websocket_command({
|
|
|
|
'type': 'cloud/remote/disconnect'
|
|
|
|
})
|
|
|
|
async def websocket_remote_disconnect(hass, connection, msg):
|
|
|
|
"""Handle request for disconnect remote."""
|
|
|
|
cloud = hass.data[DOMAIN]
|
|
|
|
await cloud.client.prefs.async_update(remote_enabled=False)
|
2019-03-16 02:26:10 +00:00
|
|
|
await cloud.remote.disconnect()
|
2019-03-12 14:54:04 +00:00
|
|
|
connection.send_result(msg['id'], _account_data(cloud))
|
2019-05-29 15:39:12 +00:00
|
|
|
|
|
|
|
|
|
|
|
@websocket_api.require_admin
|
|
|
|
@_require_cloud_login
|
|
|
|
@websocket_api.async_response
|
|
|
|
@_ws_handle_cloud_errors
|
|
|
|
@websocket_api.websocket_command({
|
|
|
|
'type': 'cloud/google_assistant/entities'
|
|
|
|
})
|
|
|
|
async def google_assistant_list(hass, connection, msg):
|
|
|
|
"""List all google assistant entities."""
|
|
|
|
cloud = hass.data[DOMAIN]
|
|
|
|
entities = google_helpers.async_get_entities(
|
|
|
|
hass, cloud.client.google_config
|
|
|
|
)
|
|
|
|
|
|
|
|
result = []
|
|
|
|
|
|
|
|
for entity in entities:
|
|
|
|
result.append({
|
|
|
|
'entity_id': entity.entity_id,
|
|
|
|
'traits': [trait.name for trait in entity.traits()],
|
|
|
|
'might_2fa': entity.might_2fa(),
|
|
|
|
})
|
|
|
|
|
|
|
|
connection.send_result(msg['id'], result)
|
|
|
|
|
|
|
|
|
|
|
|
@websocket_api.require_admin
|
|
|
|
@_require_cloud_login
|
|
|
|
@websocket_api.async_response
|
|
|
|
@_ws_handle_cloud_errors
|
|
|
|
@websocket_api.websocket_command({
|
|
|
|
'type': 'cloud/google_assistant/entities/update',
|
|
|
|
'entity_id': str,
|
|
|
|
vol.Optional('should_expose'): bool,
|
|
|
|
vol.Optional('override_name'): str,
|
|
|
|
vol.Optional('aliases'): [str],
|
|
|
|
vol.Optional('disable_2fa'): bool,
|
|
|
|
})
|
|
|
|
async def google_assistant_update(hass, connection, msg):
|
2019-06-13 15:43:57 +00:00
|
|
|
"""Update google assistant config."""
|
2019-05-29 15:39:12 +00:00
|
|
|
cloud = hass.data[DOMAIN]
|
|
|
|
changes = dict(msg)
|
|
|
|
changes.pop('type')
|
|
|
|
changes.pop('id')
|
|
|
|
|
|
|
|
await cloud.client.prefs.async_update_google_entity_config(**changes)
|
|
|
|
|
|
|
|
connection.send_result(
|
|
|
|
msg['id'],
|
|
|
|
cloud.client.prefs.google_entity_configs.get(msg['entity_id']))
|
2019-06-13 18:58:08 +00:00
|
|
|
|
|
|
|
|
|
|
|
@websocket_api.require_admin
|
|
|
|
@_require_cloud_login
|
|
|
|
@websocket_api.async_response
|
|
|
|
@_ws_handle_cloud_errors
|
|
|
|
@websocket_api.websocket_command({
|
|
|
|
'type': 'cloud/alexa/entities'
|
|
|
|
})
|
|
|
|
async def alexa_list(hass, connection, msg):
|
|
|
|
"""List all alexa entities."""
|
|
|
|
cloud = hass.data[DOMAIN]
|
|
|
|
entities = alexa_entities.async_get_entities(
|
|
|
|
hass, cloud.client.alexa_config
|
|
|
|
)
|
|
|
|
|
|
|
|
result = []
|
|
|
|
|
|
|
|
for entity in entities:
|
|
|
|
result.append({
|
|
|
|
'entity_id': entity.entity_id,
|
|
|
|
'display_categories': entity.default_display_categories(),
|
|
|
|
'interfaces': [ifc.name() for ifc in entity.interfaces()],
|
|
|
|
})
|
|
|
|
|
|
|
|
connection.send_result(msg['id'], result)
|
|
|
|
|
|
|
|
|
|
|
|
@websocket_api.require_admin
|
|
|
|
@_require_cloud_login
|
|
|
|
@websocket_api.async_response
|
|
|
|
@_ws_handle_cloud_errors
|
|
|
|
@websocket_api.websocket_command({
|
|
|
|
'type': 'cloud/alexa/entities/update',
|
|
|
|
'entity_id': str,
|
|
|
|
vol.Optional('should_expose'): bool,
|
|
|
|
})
|
|
|
|
async def alexa_update(hass, connection, msg):
|
|
|
|
"""Update alexa entity config."""
|
|
|
|
cloud = hass.data[DOMAIN]
|
|
|
|
changes = dict(msg)
|
|
|
|
changes.pop('type')
|
|
|
|
changes.pop('id')
|
|
|
|
|
|
|
|
await cloud.client.prefs.async_update_alexa_entity_config(**changes)
|
|
|
|
|
|
|
|
connection.send_result(
|
|
|
|
msg['id'],
|
|
|
|
cloud.client.prefs.alexa_entity_configs.get(msg['entity_id']))
|