* CREATE Somfy component * CREATE cover Somfy platform * USE somfy id as unique id * UPDATE all the devices in one call to limit the number of call * FIX Don't load devices if not yet configured * IMP Replace configurator by a simple notification * ADD log in case state does not match * IMP wording * REMOVE debug stuf * ADD support for tilt position * UPDATE requirements * FIX Use code instead of authorization response - Will allow to setup Somfy without https * HANDLE stateless devices (Somfy RTS) * FIX import locally 3rd party library * UPDATE pymfy to 0.4.3 * ADD missing docstring * FIX For Somfy 100 means closed and 0 opened * FIX position can be None * ENHANCE error management when error 500 occurs at setup * FIX indent * ROLLBACK tilt modification - See https://community.home-assistant.io/t/somfy-tahoma-official-api/61448/90?u=tetienne * FIX Look for capability instead of state * DON'T use exception to test if a feature is available * UPDATE dependency * ADD device_info property * AVOID object creation in each method * REMOVE unused constants * ADD missing doc * IMP Only make one call to add_entities * USE dict[key] instead of get method * IMP Don't pass hass object to the entities * FIX Don't end logging messages with period * USE config entries instead of a cache file * IMPLEMENT async_unload_entry * CONSOLIDATE package - see home-assistant/architecture#124 * UPDATE to pymfy 0.5.1 * SIMPLIFY config flow * ADD French translation * FIX 80 vs 79 max length * ABORT flow asap * FIX A tupple was returned * MIGRATE to manifest.json * ADD a placeholder async_setup_platform coroutine - It's currently required and expected by the platform helper. * FIX codeowner * ADD missing translations file * USE new external step * UPGRADE pymfy version * Close Somfy tab automatically * ADD manufacturer - Somfy only for the moment. * HANDLE missing code or state in Somfy request * REMOVE unused strings * DECLARE somfy component to use config_flow * APPLY static check remarks * FIX async method cannot be called from sync context * FIX only unload what has been loaded during entry setup * DON't catch them all * DON'T log full stacktrace * ABORT conflig flow if configuration missing * OMIT Somfy files for coverage * ADD tests about Somfy config flow * ADD pymfy to the test dependencies
146 lines
4.8 KiB
Python
146 lines
4.8 KiB
Python
"""Config flow for Somfy."""
|
|
import asyncio
|
|
import logging
|
|
|
|
import async_timeout
|
|
|
|
from homeassistant import config_entries
|
|
from homeassistant.components.http import HomeAssistantView
|
|
from homeassistant.core import callback
|
|
from .const import CLIENT_ID, CLIENT_SECRET, DOMAIN
|
|
|
|
AUTH_CALLBACK_PATH = '/auth/somfy/callback'
|
|
AUTH_CALLBACK_NAME = 'auth:somfy:callback'
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
@callback
|
|
def register_flow_implementation(hass, client_id, client_secret):
|
|
"""Register a flow implementation.
|
|
|
|
client_id: Client id.
|
|
client_secret: Client secret.
|
|
"""
|
|
hass.data[DOMAIN][CLIENT_ID] = client_id
|
|
hass.data[DOMAIN][CLIENT_SECRET] = client_secret
|
|
|
|
|
|
@config_entries.HANDLERS.register('somfy')
|
|
class SomfyFlowHandler(config_entries.ConfigFlow):
|
|
"""Handle a config flow."""
|
|
|
|
VERSION = 1
|
|
CONNECTION_CLASS = config_entries.CONN_CLASS_CLOUD_POLL
|
|
|
|
def __init__(self):
|
|
"""Instantiate config flow."""
|
|
self.code = None
|
|
|
|
async def async_step_import(self, user_input=None):
|
|
"""Handle external yaml configuration."""
|
|
if self.hass.config_entries.async_entries(DOMAIN):
|
|
return self.async_abort(reason='already_setup')
|
|
return await self.async_step_auth()
|
|
|
|
async def async_step_user(self, user_input=None):
|
|
"""Handle a flow start."""
|
|
if self.hass.config_entries.async_entries(DOMAIN):
|
|
return self.async_abort(reason='already_setup')
|
|
|
|
if DOMAIN not in self.hass.data:
|
|
return self.async_abort(reason='missing_configuration')
|
|
|
|
return await self.async_step_auth()
|
|
|
|
async def async_step_auth(self, user_input=None):
|
|
"""Create an entry for auth."""
|
|
# Flow has been triggered from Somfy website
|
|
if user_input:
|
|
return await self.async_step_code(user_input)
|
|
|
|
try:
|
|
with async_timeout.timeout(10):
|
|
url, _ = await self._get_authorization_url()
|
|
except asyncio.TimeoutError:
|
|
return self.async_abort(reason='authorize_url_timeout')
|
|
|
|
return self.async_external_step(
|
|
step_id='auth',
|
|
url=url
|
|
)
|
|
|
|
async def _get_authorization_url(self):
|
|
"""Get Somfy authorization url."""
|
|
from pymfy.api.somfy_api import SomfyApi
|
|
client_id = self.hass.data[DOMAIN][CLIENT_ID]
|
|
client_secret = self.hass.data[DOMAIN][CLIENT_SECRET]
|
|
redirect_uri = '{}{}'.format(
|
|
self.hass.config.api.base_url, AUTH_CALLBACK_PATH)
|
|
api = SomfyApi(client_id, client_secret, redirect_uri)
|
|
|
|
self.hass.http.register_view(SomfyAuthCallbackView())
|
|
# Thanks to the state, we can forward the flow id to Somfy that will
|
|
# add it in the callback.
|
|
return await self.hass.async_add_executor_job(
|
|
api.get_authorization_url, self.flow_id)
|
|
|
|
async def async_step_code(self, code):
|
|
"""Received code for authentication."""
|
|
self.code = code
|
|
return self.async_external_step_done(next_step_id="creation")
|
|
|
|
async def async_step_creation(self, user_input=None):
|
|
"""Create Somfy api and entries."""
|
|
client_id = self.hass.data[DOMAIN][CLIENT_ID]
|
|
client_secret = self.hass.data[DOMAIN][CLIENT_SECRET]
|
|
code = self.code
|
|
from pymfy.api.somfy_api import SomfyApi
|
|
redirect_uri = '{}{}'.format(
|
|
self.hass.config.api.base_url, AUTH_CALLBACK_PATH)
|
|
api = SomfyApi(client_id, client_secret, redirect_uri)
|
|
token = await self.hass.async_add_executor_job(api.request_token, None,
|
|
code)
|
|
_LOGGER.info('Successfully authenticated Somfy')
|
|
return self.async_create_entry(
|
|
title='Somfy',
|
|
data={
|
|
'token': token,
|
|
'refresh_args': {
|
|
'client_id': client_id,
|
|
'client_secret': client_secret
|
|
}
|
|
},
|
|
)
|
|
|
|
|
|
class SomfyAuthCallbackView(HomeAssistantView):
|
|
"""Somfy Authorization Callback View."""
|
|
|
|
requires_auth = False
|
|
url = AUTH_CALLBACK_PATH
|
|
name = AUTH_CALLBACK_NAME
|
|
|
|
@staticmethod
|
|
async def get(request):
|
|
"""Receive authorization code."""
|
|
from aiohttp import web_response
|
|
|
|
if 'code' not in request.query or 'state' not in request.query:
|
|
return web_response.Response(
|
|
text="Missing code or state parameter in " + request.url
|
|
)
|
|
|
|
hass = request.app['hass']
|
|
hass.async_create_task(
|
|
hass.config_entries.flow.async_configure(
|
|
flow_id=request.query['state'],
|
|
user_input=request.query['code'],
|
|
))
|
|
|
|
return web_response.Response(
|
|
headers={
|
|
'content-type': 'text/html'
|
|
},
|
|
text="<script>window.close()</script>"
|
|
)
|