2023-01-18 16:33:15 +00:00
|
|
|
"""Websocket API for OTBR."""
|
|
|
|
from typing import TYPE_CHECKING
|
|
|
|
|
2023-02-27 15:19:13 +00:00
|
|
|
import python_otbr_api
|
|
|
|
|
2023-03-08 18:21:04 +00:00
|
|
|
from homeassistant.components import websocket_api
|
2023-01-18 16:33:15 +00:00
|
|
|
from homeassistant.core import HomeAssistant, callback
|
|
|
|
from homeassistant.exceptions import HomeAssistantError
|
|
|
|
|
2023-02-28 16:08:45 +00:00
|
|
|
from .const import DEFAULT_CHANNEL, DOMAIN
|
2023-01-18 16:33:15 +00:00
|
|
|
|
|
|
|
if TYPE_CHECKING:
|
|
|
|
from . import OTBRData
|
|
|
|
|
|
|
|
|
|
|
|
@callback
|
2023-01-24 05:55:12 +00:00
|
|
|
def async_setup(hass: HomeAssistant) -> None:
|
2023-01-18 16:33:15 +00:00
|
|
|
"""Set up the OTBR Websocket API."""
|
2023-03-08 18:21:04 +00:00
|
|
|
websocket_api.async_register_command(hass, websocket_info)
|
|
|
|
websocket_api.async_register_command(hass, websocket_create_network)
|
2023-01-18 16:33:15 +00:00
|
|
|
|
|
|
|
|
2023-03-08 18:21:04 +00:00
|
|
|
@websocket_api.websocket_command(
|
2023-01-18 16:33:15 +00:00
|
|
|
{
|
|
|
|
"type": "otbr/info",
|
|
|
|
}
|
|
|
|
)
|
2023-03-08 18:21:04 +00:00
|
|
|
@websocket_api.require_admin
|
|
|
|
@websocket_api.async_response
|
2023-01-18 16:33:15 +00:00
|
|
|
async def websocket_info(
|
2023-03-08 18:21:04 +00:00
|
|
|
hass: HomeAssistant, connection: websocket_api.ActiveConnection, msg: dict
|
2023-01-18 16:33:15 +00:00
|
|
|
) -> None:
|
|
|
|
"""Get OTBR info."""
|
|
|
|
if DOMAIN not in hass.data:
|
|
|
|
connection.send_error(msg["id"], "not_loaded", "No OTBR API loaded")
|
|
|
|
return
|
|
|
|
|
|
|
|
data: OTBRData = hass.data[DOMAIN]
|
|
|
|
|
|
|
|
try:
|
|
|
|
dataset = await data.get_active_dataset_tlvs()
|
|
|
|
except HomeAssistantError as exc:
|
|
|
|
connection.send_error(msg["id"], "get_dataset_failed", str(exc))
|
|
|
|
return
|
|
|
|
|
|
|
|
connection.send_result(
|
|
|
|
msg["id"],
|
|
|
|
{
|
|
|
|
"url": data.url,
|
2023-01-24 05:55:12 +00:00
|
|
|
"active_dataset_tlvs": dataset.hex() if dataset else None,
|
2023-01-18 16:33:15 +00:00
|
|
|
},
|
|
|
|
)
|
2023-02-27 15:19:13 +00:00
|
|
|
|
|
|
|
|
2023-03-08 18:21:04 +00:00
|
|
|
@websocket_api.websocket_command(
|
2023-02-27 15:19:13 +00:00
|
|
|
{
|
|
|
|
"type": "otbr/create_network",
|
|
|
|
}
|
|
|
|
)
|
2023-03-08 18:21:04 +00:00
|
|
|
@websocket_api.require_admin
|
|
|
|
@websocket_api.async_response
|
2023-02-27 15:19:13 +00:00
|
|
|
async def websocket_create_network(
|
2023-03-08 18:21:04 +00:00
|
|
|
hass: HomeAssistant, connection: websocket_api.ActiveConnection, msg: dict
|
2023-02-27 15:19:13 +00:00
|
|
|
) -> None:
|
|
|
|
"""Create a new Thread network."""
|
|
|
|
if DOMAIN not in hass.data:
|
|
|
|
connection.send_error(msg["id"], "not_loaded", "No OTBR API loaded")
|
|
|
|
return
|
|
|
|
|
2023-02-28 16:08:45 +00:00
|
|
|
# We currently have no way to know which channel zha is using, assume it's
|
|
|
|
# the default
|
|
|
|
zha_channel = DEFAULT_CHANNEL
|
|
|
|
|
2023-02-27 15:19:13 +00:00
|
|
|
data: OTBRData = hass.data[DOMAIN]
|
|
|
|
|
|
|
|
try:
|
|
|
|
await data.set_enabled(False)
|
|
|
|
except HomeAssistantError as exc:
|
|
|
|
connection.send_error(msg["id"], "set_enabled_failed", str(exc))
|
|
|
|
return
|
|
|
|
|
|
|
|
try:
|
|
|
|
await data.create_active_dataset(
|
2023-02-28 16:08:45 +00:00
|
|
|
python_otbr_api.OperationalDataSet(
|
|
|
|
channel=zha_channel, network_name="home-assistant"
|
|
|
|
)
|
2023-02-27 15:19:13 +00:00
|
|
|
)
|
|
|
|
except HomeAssistantError as exc:
|
|
|
|
connection.send_error(msg["id"], "create_active_dataset_failed", str(exc))
|
|
|
|
return
|
|
|
|
|
|
|
|
try:
|
|
|
|
await data.set_enabled(True)
|
|
|
|
except HomeAssistantError as exc:
|
|
|
|
connection.send_error(msg["id"], "set_enabled_failed", str(exc))
|
|
|
|
return
|
|
|
|
|
|
|
|
connection.send_result(msg["id"])
|