2024-04-23 16:18:57 +00:00
|
|
|
"""Implementation for child device setup.
|
|
|
|
|
|
|
|
This module allows pairing and disconnecting child devices.
|
|
|
|
"""
|
|
|
|
|
|
|
|
from __future__ import annotations
|
|
|
|
|
|
|
|
import asyncio
|
|
|
|
import logging
|
2024-05-04 13:34:54 +00:00
|
|
|
from typing import TYPE_CHECKING
|
2024-04-23 16:18:57 +00:00
|
|
|
|
|
|
|
# hen support for cpython older than 3.11 is dropped
|
|
|
|
# async_timeout can be replaced with asyncio.timeout
|
|
|
|
from async_timeout import timeout as asyncio_timeout
|
|
|
|
|
2024-04-24 16:57:35 +00:00
|
|
|
from ...feature import Feature
|
2024-04-23 16:18:57 +00:00
|
|
|
from ..smartmodule import SmartModule
|
|
|
|
|
2024-05-04 13:34:54 +00:00
|
|
|
if TYPE_CHECKING:
|
|
|
|
from ..smartdevice import SmartDevice
|
|
|
|
|
2024-04-23 16:18:57 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
2024-06-15 00:43:48 +00:00
|
|
|
class ChildSetup(SmartModule):
|
2024-04-23 16:18:57 +00:00
|
|
|
"""Implementation for child device setup."""
|
|
|
|
|
|
|
|
REQUIRED_COMPONENT = "child_quick_setup"
|
|
|
|
QUERY_GETTER_NAME = "get_support_child_device_category"
|
|
|
|
|
2024-04-24 16:57:35 +00:00
|
|
|
def __init__(self, device: SmartDevice, module: str):
|
|
|
|
super().__init__(device, module)
|
|
|
|
self._add_feature(
|
|
|
|
Feature(
|
|
|
|
device,
|
2024-06-15 00:43:48 +00:00
|
|
|
id="pair",
|
2024-04-24 16:57:35 +00:00
|
|
|
name="Pair",
|
|
|
|
container=self,
|
|
|
|
attribute_setter="pair",
|
|
|
|
category=Feature.Category.Config,
|
|
|
|
type=Feature.Type.Action,
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
2024-04-23 16:18:57 +00:00
|
|
|
@property
|
|
|
|
def supported_device_categories(self) -> list[str]:
|
|
|
|
"""Return supported device categories."""
|
|
|
|
return self.data["device_category_list"]
|
|
|
|
|
2024-04-24 16:57:35 +00:00
|
|
|
async def pair(self, *, timeout=10):
|
2024-04-23 16:18:57 +00:00
|
|
|
"""Scan for new devices and pair after discovering first new device."""
|
|
|
|
await self.call("begin_scanning_child_device")
|
|
|
|
|
|
|
|
discovered: dict = {}
|
|
|
|
try:
|
|
|
|
async with asyncio_timeout(timeout):
|
|
|
|
while True:
|
|
|
|
await asyncio.sleep(0.5)
|
|
|
|
res = await self.get_detected_devices()
|
|
|
|
if res["child_device_list"]:
|
|
|
|
discovered = res
|
|
|
|
break
|
|
|
|
|
|
|
|
except TimeoutError:
|
|
|
|
pass
|
|
|
|
|
2024-04-24 16:57:35 +00:00
|
|
|
if not discovered:
|
|
|
|
_LOGGER.warning("No devices found.")
|
|
|
|
return
|
|
|
|
|
2024-04-23 16:18:57 +00:00
|
|
|
_LOGGER.info(
|
|
|
|
"Discovery done, found %s devices", len(discovered["child_device_list"])
|
|
|
|
)
|
2024-04-24 16:57:35 +00:00
|
|
|
|
|
|
|
return await self.add_devices(discovered)
|
2024-04-23 16:18:57 +00:00
|
|
|
|
|
|
|
async def unpair(self, device_id: str):
|
|
|
|
"""Remove device from the hub."""
|
|
|
|
payload = {"child_device_list": [{"device_id": device_id}]}
|
2024-05-04 13:34:54 +00:00
|
|
|
res = await self._device._query_helper("remove_child_device_list", payload)
|
2024-06-15 00:43:48 +00:00
|
|
|
self._device.request_renegotiation()
|
2024-05-04 13:34:54 +00:00
|
|
|
return res
|
2024-04-23 16:18:57 +00:00
|
|
|
|
|
|
|
async def add_devices(self, devices: dict):
|
|
|
|
"""Add devices."""
|
2024-05-04 13:34:54 +00:00
|
|
|
res = await self._device._query_helper("add_child_device_list", devices)
|
2024-06-15 00:43:48 +00:00
|
|
|
self._device.request_renegotiation()
|
2024-05-04 13:34:54 +00:00
|
|
|
return res
|
2024-04-23 16:18:57 +00:00
|
|
|
|
|
|
|
async def get_detected_devices(self) -> dict:
|
|
|
|
"""Return list of devices detected during scanning."""
|
|
|
|
param = {"scan_list": self.supported_device_categories}
|
|
|
|
res = await self.call("get_scan_child_device_list", param)
|
|
|
|
_LOGGER.debug("Scan status: %s", res)
|
|
|
|
return res["get_scan_child_device_list"]
|