Home Assistant Unofficial Reference 2024.12.1
__init__.py
Go to the documentation of this file.
1 """The Risco integration."""
2 
3 from __future__ import annotations
4 
5 from collections.abc import Callable
6 from dataclasses import dataclass, field
7 import logging
8 from typing import Any
9 
10 from pyrisco import CannotConnectError, RiscoCloud, RiscoLocal, UnauthorizedError
11 from pyrisco.common import Partition, System, Zone
12 
13 from homeassistant.config_entries import ConfigEntry
14 from homeassistant.const import (
15  CONF_HOST,
16  CONF_PASSWORD,
17  CONF_PIN,
18  CONF_PORT,
19  CONF_SCAN_INTERVAL,
20  CONF_TYPE,
21  CONF_USERNAME,
22  Platform,
23 )
24 from homeassistant.core import HomeAssistant
25 from homeassistant.exceptions import ConfigEntryAuthFailed, ConfigEntryNotReady
26 from homeassistant.helpers.aiohttp_client import async_get_clientsession
27 from homeassistant.helpers.dispatcher import async_dispatcher_send
28 
29 from .const import (
30  CONF_CONCURRENCY,
31  DATA_COORDINATOR,
32  DEFAULT_CONCURRENCY,
33  DEFAULT_SCAN_INTERVAL,
34  DOMAIN,
35  EVENTS_COORDINATOR,
36  SYSTEM_UPDATE_SIGNAL,
37  TYPE_LOCAL,
38 )
39 from .coordinator import RiscoDataUpdateCoordinator, RiscoEventsDataUpdateCoordinator
40 
41 PLATFORMS = [
42  Platform.ALARM_CONTROL_PANEL,
43  Platform.BINARY_SENSOR,
44  Platform.SENSOR,
45  Platform.SWITCH,
46 ]
47 _LOGGER = logging.getLogger(__name__)
48 
49 
50 @dataclass
51 class LocalData:
52  """A data class for local data passed to the platforms."""
53 
54  system: RiscoLocal
55  partition_updates: dict[int, Callable[[], Any]] = field(default_factory=dict)
56 
57 
58 def is_local(entry: ConfigEntry) -> bool:
59  """Return whether the entry represents an instance with local communication."""
60  return entry.data.get(CONF_TYPE) == TYPE_LOCAL
61 
62 
63 def zone_update_signal(zone_id: int) -> str:
64  """Return a signal for the dispatch of a zone update."""
65  return f"risco_zone_update_{zone_id}"
66 
67 
68 async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
69  """Set up Risco from a config entry."""
70  if is_local(entry):
71  return await _async_setup_local_entry(hass, entry)
72 
73  return await _async_setup_cloud_entry(hass, entry)
74 
75 
76 async def _async_setup_local_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
77  data = entry.data
78  concurrency = entry.options.get(CONF_CONCURRENCY, DEFAULT_CONCURRENCY)
79  risco = RiscoLocal(
80  data[CONF_HOST], data[CONF_PORT], data[CONF_PIN], concurrency=concurrency
81  )
82 
83  try:
84  await risco.connect()
85  except CannotConnectError as error:
86  raise ConfigEntryNotReady from error
87  except UnauthorizedError:
88  _LOGGER.exception("Failed to login to Risco cloud")
89  return False
90 
91  async def _error(error: Exception) -> None:
92  _LOGGER.error("Error in Risco library", exc_info=error)
93  if isinstance(error, ConnectionResetError) and not hass.is_stopping:
94  _LOGGER.debug("Disconnected from panel. Reloading integration")
95  hass.async_create_task(hass.config_entries.async_reload(entry.entry_id))
96 
97  entry.async_on_unload(risco.add_error_handler(_error))
98 
99  async def _default(command: str, result: str, *params: list[str]) -> None:
100  _LOGGER.debug(
101  "Unhandled update from Risco library: %s, %s, %s", command, result, params
102  )
103 
104  entry.async_on_unload(risco.add_default_handler(_default))
105 
106  local_data = LocalData(risco)
107 
108  async def _zone(zone_id: int, zone: Zone) -> None:
109  _LOGGER.debug("Risco zone update for %d", zone_id)
111 
112  entry.async_on_unload(risco.add_zone_handler(_zone))
113 
114  async def _partition(partition_id: int, partition: Partition) -> None:
115  _LOGGER.debug("Risco partition update for %d", partition_id)
116  callback = local_data.partition_updates.get(partition_id)
117  if callback:
118  callback()
119 
120  entry.async_on_unload(risco.add_partition_handler(_partition))
121 
122  async def _system(system: System) -> None:
123  _LOGGER.debug("Risco system update")
124  async_dispatcher_send(hass, SYSTEM_UPDATE_SIGNAL)
125 
126  entry.async_on_unload(risco.add_system_handler(_system))
127 
128  entry.async_on_unload(entry.add_update_listener(_update_listener))
129 
130  hass.data.setdefault(DOMAIN, {})
131  hass.data[DOMAIN][entry.entry_id] = local_data
132  await hass.config_entries.async_forward_entry_setups(entry, PLATFORMS)
133 
134  return True
135 
136 
137 async def _async_setup_cloud_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
138  data = entry.data
139  risco = RiscoCloud(data[CONF_USERNAME], data[CONF_PASSWORD], data[CONF_PIN])
140  try:
141  await risco.login(async_get_clientsession(hass))
142  except CannotConnectError as error:
143  raise ConfigEntryNotReady from error
144  except UnauthorizedError as error:
145  raise ConfigEntryAuthFailed from error
146 
147  scan_interval = entry.options.get(CONF_SCAN_INTERVAL, DEFAULT_SCAN_INTERVAL)
148  coordinator = RiscoDataUpdateCoordinator(hass, risco, scan_interval)
149  await coordinator.async_config_entry_first_refresh()
150  events_coordinator = RiscoEventsDataUpdateCoordinator(
151  hass, risco, entry.entry_id, 60
152  )
153 
154  entry.async_on_unload(entry.add_update_listener(_update_listener))
155 
156  hass.data.setdefault(DOMAIN, {})
157  hass.data[DOMAIN][entry.entry_id] = {
158  DATA_COORDINATOR: coordinator,
159  EVENTS_COORDINATOR: events_coordinator,
160  }
161 
162  await hass.config_entries.async_forward_entry_setups(entry, PLATFORMS)
163  await events_coordinator.async_refresh()
164 
165  return True
166 
167 
168 async def async_unload_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
169  """Unload a config entry."""
170  unload_ok = await hass.config_entries.async_unload_platforms(entry, PLATFORMS)
171  if unload_ok:
172  if is_local(entry):
173  local_data: LocalData = hass.data[DOMAIN][entry.entry_id]
174  await local_data.system.disconnect()
175 
176  hass.data[DOMAIN].pop(entry.entry_id)
177 
178  return unload_ok
179 
180 
181 async def _update_listener(hass: HomeAssistant, entry: ConfigEntry) -> None:
182  """Handle options update."""
183  await hass.config_entries.async_reload(entry.entry_id)
None _update_listener(HomeAssistant hass, ConfigEntry entry)
Definition: __init__.py:181
bool _async_setup_local_entry(HomeAssistant hass, ConfigEntry entry)
Definition: __init__.py:76
bool async_unload_entry(HomeAssistant hass, ConfigEntry entry)
Definition: __init__.py:168
bool async_setup_entry(HomeAssistant hass, ConfigEntry entry)
Definition: __init__.py:68
str zone_update_signal(int zone_id)
Definition: __init__.py:63
bool _async_setup_cloud_entry(HomeAssistant hass, ConfigEntry entry)
Definition: __init__.py:137
bool is_local(ConfigEntry entry)
Definition: __init__.py:58
aiohttp.ClientSession async_get_clientsession(HomeAssistant hass, bool verify_ssl=True, socket.AddressFamily family=socket.AF_UNSPEC, ssl_util.SSLCipherList ssl_cipher=ssl_util.SSLCipherList.PYTHON_DEFAULT)
None async_dispatcher_send(HomeAssistant hass, str signal, *Any args)
Definition: dispatcher.py:193