Home Assistant Unofficial Reference 2024.12.1
config_flow.py
Go to the documentation of this file.
1 """Adds config flow for MicroBot."""
2 
3 from __future__ import annotations
4 
5 import logging
6 from typing import Any
7 
8 from bleak.backends.device import BLEDevice
9 from microbot import (
10  MicroBotAdvertisement,
11  MicroBotApiClient,
12  parse_advertisement_data,
13  randomid,
14 )
15 import voluptuous as vol
16 
18  BluetoothServiceInfoBleak,
19  async_discovered_service_info,
20 )
21 from homeassistant.config_entries import ConfigFlow, ConfigFlowResult
22 from homeassistant.const import CONF_ACCESS_TOKEN, CONF_ADDRESS
23 
24 from .const import DOMAIN
25 
26 _LOGGER: logging.Logger = logging.getLogger(__package__)
27 
28 
29 def short_address(address: str) -> str:
30  """Convert a Bluetooth address to a short address."""
31  results = address.replace("-", ":").split(":")
32  return f"{results[0].upper()}{results[1].upper()}"[0:4]
33 
34 
35 def name_from_discovery(discovery: MicroBotAdvertisement) -> str:
36  """Get the name from a discovery."""
37  return f'{discovery.data["local_name"]} {short_address(discovery.address)}'
38 
39 
40 class MicroBotConfigFlow(ConfigFlow, domain=DOMAIN):
41  """Config flow for MicroBot."""
42 
43  VERSION = 1
44 
45  def __init__(self) -> None:
46  """Initialize."""
47  self._errors: dict[str, str] = {}
48  self._discovered_adv_discovered_adv: MicroBotAdvertisement | None = None
49  self._discovered_advs: dict[str, MicroBotAdvertisement] = {}
50  self._client_client: MicroBotApiClient | None = None
51  self._ble_device_ble_device: BLEDevice | None = None
52  self._name_name: str | None = None
53  self._bdaddr_bdaddr: str | None = None
54 
56  self, discovery_info: BluetoothServiceInfoBleak
57  ) -> ConfigFlowResult:
58  """Handle the bluetooth discovery step."""
59  _LOGGER.debug("Discovered bluetooth device: %s", discovery_info)
60  await self.async_set_unique_idasync_set_unique_id(discovery_info.address)
61  self._abort_if_unique_id_configured_abort_if_unique_id_configured()
62  self._ble_device_ble_device = discovery_info.device
63  parsed = parse_advertisement_data(
64  discovery_info.device, discovery_info.advertisement
65  )
66  self._discovered_adv_discovered_adv = parsed
67  self.context["title_placeholders"] = {
68  "name": name_from_discovery(self._discovered_adv_discovered_adv),
69  }
70  return await self.async_step_initasync_step_init()
71 
72  async def async_step_user(
73  self, user_input: dict[str, Any] | None = None
74  ) -> ConfigFlowResult:
75  """Handle a flow initialized by the user."""
76  # This is for backwards compatibility.
77  return await self.async_step_initasync_step_init(user_input)
78 
79  async def async_step_init(
80  self, user_input: dict[str, Any] | None = None
81  ) -> ConfigFlowResult:
82  """Check if paired."""
83  errors: dict[str, str] = {}
84 
85  if discovery := self._discovered_adv_discovered_adv:
86  self._discovered_advs[discovery.address] = discovery
87  else:
88  current_addresses = self._async_current_ids_async_current_ids()
89  for discovery_info in async_discovered_service_info(self.hass):
90  self._ble_device_ble_device = discovery_info.device
91  address = discovery_info.address
92  if address in current_addresses or address in self._discovered_advs:
93  continue
94  parsed = parse_advertisement_data(
95  discovery_info.device, discovery_info.advertisement
96  )
97  if parsed:
98  self._discovered_adv_discovered_adv = parsed
99  self._discovered_advs[address] = parsed
100 
101  if not self._discovered_advs:
102  return self.async_abortasync_abortasync_abort(reason="no_devices_found")
103 
104  if user_input is not None:
105  self._name_name = name_from_discovery(self._discovered_adv_discovered_adv)
106  self._bdaddr_bdaddr = user_input[CONF_ADDRESS]
107  await self.async_set_unique_idasync_set_unique_id(self._bdaddr_bdaddr, raise_on_progress=False)
108  self._abort_if_unique_id_configured_abort_if_unique_id_configured()
109  return await self.async_step_linkasync_step_link()
110 
111  return self.async_show_formasync_show_formasync_show_form(
112  step_id="init",
113  data_schema=vol.Schema(
114  {
115  vol.Required(CONF_ADDRESS): vol.In(
116  {
117  address: f"{parsed.data['local_name']} ({address})"
118  for address, parsed in self._discovered_advs.items()
119  }
120  )
121  }
122  ),
123  errors=errors,
124  )
125 
126  async def async_step_link(
127  self, user_input: dict[str, Any] | None = None
128  ) -> ConfigFlowResult:
129  """Given a configured host, will ask the user to press the button to pair."""
130  errors: dict[str, str] = {}
131  token = randomid(32)
132  self._client_client = MicroBotApiClient(
133  device=self._ble_device_ble_device,
134  token=token,
135  )
136  assert self._client_client is not None
137  if user_input is None:
138  await self._client_client.connect(init=True)
139  return self.async_show_formasync_show_formasync_show_form(step_id="link")
140 
141  if not await self._client_client.is_connected():
142  await self._client_client.connect(init=False)
143  if not await self._client_client.is_connected():
144  errors["base"] = "linking"
145  else:
146  await self._client_client.disconnect()
147 
148  if errors:
149  return self.async_show_formasync_show_formasync_show_form(step_id="link", errors=errors)
150 
151  assert self._name_name is not None
152  return self.async_create_entryasync_create_entryasync_create_entry(
153  title=self._name_name,
154  data=user_input
155  | {
156  CONF_ADDRESS: self._bdaddr_bdaddr,
157  CONF_ACCESS_TOKEN: token,
158  },
159  )
ConfigFlowResult async_step_user(self, dict[str, Any]|None user_input=None)
Definition: config_flow.py:74
ConfigFlowResult async_step_init(self, dict[str, Any]|None user_input=None)
Definition: config_flow.py:81
ConfigFlowResult async_step_bluetooth(self, BluetoothServiceInfoBleak discovery_info)
Definition: config_flow.py:57
ConfigFlowResult async_step_link(self, dict[str, Any]|None user_input=None)
Definition: config_flow.py:128
None _abort_if_unique_id_configured(self, dict[str, Any]|None updates=None, bool reload_on_update=True, *str error="already_configured")
set[str|None] _async_current_ids(self, bool include_ignore=True)
ConfigEntry|None async_set_unique_id(self, str|None unique_id=None, *bool raise_on_progress=True)
ConfigFlowResult async_create_entry(self, *str title, Mapping[str, Any] data, str|None description=None, Mapping[str, str]|None description_placeholders=None, Mapping[str, Any]|None options=None)
ConfigFlowResult async_abort(self, *str reason, Mapping[str, str]|None description_placeholders=None)
ConfigFlowResult async_show_form(self, *str|None step_id=None, vol.Schema|None data_schema=None, dict[str, str]|None errors=None, Mapping[str, str]|None description_placeholders=None, bool|None last_step=None, str|None preview=None)
_FlowResultT async_show_form(self, *str|None step_id=None, vol.Schema|None data_schema=None, dict[str, str]|None errors=None, Mapping[str, str]|None description_placeholders=None, bool|None last_step=None, str|None preview=None)
_FlowResultT async_create_entry(self, *str|None title=None, Mapping[str, Any] data, str|None description=None, Mapping[str, str]|None description_placeholders=None)
_FlowResultT async_abort(self, *str reason, Mapping[str, str]|None description_placeholders=None)
Iterable[BluetoothServiceInfoBleak] async_discovered_service_info(HomeAssistant hass, bool connectable=True)
Definition: api.py:72
str name_from_discovery(MicroBotAdvertisement discovery)
Definition: config_flow.py:35