"""Repairs implementation for supervisor integration.""" from collections.abc import Callable from types import MethodType from typing import Any import voluptuous as vol from homeassistant.components.repairs import RepairsFlow from homeassistant.core import HomeAssistant from homeassistant.data_entry_flow import FlowResult from .const import DATA_KEY_SUPERVISOR_ISSUES, PLACEHOLDER_KEY_REFERENCE from .handler import HassioAPIError, async_apply_suggestion from .issues import Issue, Suggestion, SupervisorIssues SUGGESTION_CONFIRMATION_REQUIRED = {"system_execute_reboot"} class SupervisorIssueRepairFlow(RepairsFlow): """Handler for an issue fixing flow.""" _data: dict[str, Any] | None = None _issue: Issue | None = None def __init__(self, issue_id: str) -> None: """Initialize repair flow.""" self._issue_id = issue_id super().__init__() @property def issue(self) -> Issue | None: """Get associated issue.""" if not self._issue: supervisor_issues: SupervisorIssues = self.hass.data[ DATA_KEY_SUPERVISOR_ISSUES ] self._issue = supervisor_issues.get_issue(self._issue_id) return self._issue @property def description_placeholders(self) -> dict[str, str] | None: """Get description placeholders for steps.""" return ( {PLACEHOLDER_KEY_REFERENCE: self.issue.reference} if self.issue and self.issue.reference else None ) def _async_form_for_suggestion(self, suggestion: Suggestion) -> FlowResult: """Return form for suggestion.""" return self.async_show_form( step_id=suggestion.key, data_schema=vol.Schema({}), description_placeholders=self.description_placeholders, last_step=True, ) async def async_step_init(self, _: None = None) -> FlowResult: """Handle the first step of a fix flow.""" # Out of sync with supervisor, issue is resolved or not fixable. Remove it if not self.issue or not self.issue.suggestions: return self.async_create_entry(data={}) # All suggestions have the same logic: Apply them in supervisor, # optionally with a confirmation step. Generating the required handler for each # allows for shared logic but screens can still be translated per step id. for suggestion in self.issue.suggestions: setattr( self, f"async_step_{suggestion.key}", MethodType(self._async_step(suggestion), self), ) if len(self.issue.suggestions) > 1: return self.async_show_menu( step_id="fix_menu", menu_options=[suggestion.key for suggestion in self.issue.suggestions], description_placeholders=self.description_placeholders, ) # Always show a form for one suggestion to explain to user what's happening return self._async_form_for_suggestion(self.issue.suggestions[0]) async def _async_step_apply_suggestion( self, suggestion: Suggestion, confirmed: bool = False ) -> FlowResult: """Handle applying a suggestion as a flow step. Optionally request confirmation.""" if not confirmed and suggestion.key in SUGGESTION_CONFIRMATION_REQUIRED: return self._async_form_for_suggestion(suggestion) try: await async_apply_suggestion(self.hass, suggestion.uuid) except HassioAPIError: return self.async_abort(reason="apply_suggestion_fail") return self.async_create_entry(data={}) @staticmethod def _async_step(suggestion: Suggestion) -> Callable: """Generate a step handler for a suggestion.""" async def _async_step( self: SupervisorIssueRepairFlow, user_input: dict[str, str] | None = None ) -> FlowResult: """Handle a flow step for a suggestion.""" # pylint: disable-next=protected-access return await self._async_step_apply_suggestion( suggestion, confirmed=user_input is not None ) return _async_step async def async_create_fix_flow( hass: HomeAssistant, issue_id: str, data: dict[str, str | int | float | None] | None, ) -> RepairsFlow: """Create flow.""" return SupervisorIssueRepairFlow(issue_id)