"""Config flow for YouTube integration.""" from __future__ import annotations from collections.abc import Mapping import logging from typing import Any import voluptuous as vol from youtubeaio.helper import first from youtubeaio.types import AuthScope, ForbiddenError from youtubeaio.youtube import YouTube from homeassistant.config_entries import ( ConfigEntry, ConfigFlowResult, OptionsFlowWithConfigEntry, ) from homeassistant.const import CONF_ACCESS_TOKEN, CONF_TOKEN from homeassistant.core import callback from homeassistant.helpers import config_entry_oauth2_flow from homeassistant.helpers.aiohttp_client import async_get_clientsession from homeassistant.helpers.selector import ( SelectOptionDict, SelectSelector, SelectSelectorConfig, ) from .const import ( CHANNEL_CREATION_HELP_URL, CONF_CHANNELS, DEFAULT_ACCESS, DOMAIN, LOGGER, ) class OAuth2FlowHandler( config_entry_oauth2_flow.AbstractOAuth2FlowHandler, domain=DOMAIN ): """Config flow to handle Google OAuth2 authentication.""" _data: dict[str, Any] = {} _title: str = "" DOMAIN = DOMAIN reauth_entry: ConfigEntry | None = None _youtube: YouTube | None = None @staticmethod @callback def async_get_options_flow( config_entry: ConfigEntry, ) -> YouTubeOptionsFlowHandler: """Get the options flow for this handler.""" return YouTubeOptionsFlowHandler(config_entry) @property def logger(self) -> logging.Logger: """Return logger.""" return logging.getLogger(__name__) @property def extra_authorize_data(self) -> dict[str, Any]: """Extra data that needs to be appended to the authorize url.""" return { "scope": " ".join(DEFAULT_ACCESS), # Add params to ensure we get back a refresh token "access_type": "offline", "prompt": "consent", } async def async_step_reauth( self, entry_data: Mapping[str, Any] ) -> ConfigFlowResult: """Perform reauth upon an API authentication error.""" self.reauth_entry = self.hass.config_entries.async_get_entry( self.context["entry_id"] ) return await self.async_step_reauth_confirm() async def async_step_reauth_confirm( self, user_input: dict[str, Any] | None = None ) -> ConfigFlowResult: """Confirm reauth dialog.""" if user_input is None: return self.async_show_form(step_id="reauth_confirm") return await self.async_step_user() async def get_resource(self, token: str) -> YouTube: """Get Youtube resource async.""" if self._youtube is None: self._youtube = YouTube(session=async_get_clientsession(self.hass)) await self._youtube.set_user_authentication(token, [AuthScope.READ_ONLY]) return self._youtube async def async_oauth_create_entry(self, data: dict[str, Any]) -> ConfigFlowResult: """Create an entry for the flow, or update existing entry.""" try: youtube = await self.get_resource(data[CONF_TOKEN][CONF_ACCESS_TOKEN]) own_channel = await first(youtube.get_user_channels()) if own_channel is None or own_channel.snippet is None: return self.async_abort( reason="no_channel", description_placeholders={"support_url": CHANNEL_CREATION_HELP_URL}, ) except ForbiddenError as ex: error = ex.args[0] return self.async_abort( reason="access_not_configured", description_placeholders={"message": error}, ) except Exception as ex: # pylint: disable=broad-except LOGGER.error("Unknown error occurred: %s", ex.args) return self.async_abort(reason="unknown") self._title = own_channel.snippet.title self._data = data if not self.reauth_entry: await self.async_set_unique_id(own_channel.channel_id) self._abort_if_unique_id_configured() return await self.async_step_channels() if self.reauth_entry.unique_id == own_channel.channel_id: self.hass.config_entries.async_update_entry(self.reauth_entry, data=data) await self.hass.config_entries.async_reload(self.reauth_entry.entry_id) return self.async_abort(reason="reauth_successful") return self.async_abort( reason="wrong_account", description_placeholders={"title": self._title}, ) async def async_step_channels( self, user_input: dict[str, Any] | None = None ) -> ConfigFlowResult: """Select which channels to track.""" if user_input: return self.async_create_entry( title=self._title, data=self._data, options=user_input, ) youtube = await self.get_resource(self._data[CONF_TOKEN][CONF_ACCESS_TOKEN]) selectable_channels = [ SelectOptionDict( value=subscription.snippet.channel_id, label=subscription.snippet.title, ) async for subscription in youtube.get_user_subscriptions() ] if not selectable_channels: return self.async_abort(reason="no_subscriptions") return self.async_show_form( step_id="channels", data_schema=vol.Schema( { vol.Required(CONF_CHANNELS): SelectSelector( SelectSelectorConfig(options=selectable_channels, multiple=True) ), } ), ) class YouTubeOptionsFlowHandler(OptionsFlowWithConfigEntry): """YouTube Options flow handler.""" async def async_step_init( self, user_input: dict[str, Any] | None = None ) -> ConfigFlowResult: """Initialize form.""" if user_input is not None: return self.async_create_entry( title=self.config_entry.title, data=user_input, ) youtube = YouTube(session=async_get_clientsession(self.hass)) await youtube.set_user_authentication( self.config_entry.data[CONF_TOKEN][CONF_ACCESS_TOKEN], [AuthScope.READ_ONLY] ) selectable_channels = [ SelectOptionDict( value=subscription.snippet.channel_id, label=subscription.snippet.title, ) async for subscription in youtube.get_user_subscriptions() ] return self.async_show_form( step_id="init", data_schema=self.add_suggested_values_to_schema( vol.Schema( { vol.Required(CONF_CHANNELS): SelectSelector( SelectSelectorConfig( options=selectable_channels, multiple=True ) ), } ), self.options, ), )