mirror of
https://github.com/python-kasa/python-kasa.git
synced 2025-01-22 20:57:07 +00:00
165 lines
5.5 KiB
Python
165 lines
5.5 KiB
Python
"""Module for cli discovery commands."""
|
|
|
|
from __future__ import annotations
|
|
|
|
import asyncio
|
|
from pprint import pformat as pf
|
|
|
|
import asyncclick as click
|
|
from pydantic.v1 import ValidationError
|
|
|
|
from kasa import (
|
|
AuthenticationError,
|
|
Credentials,
|
|
Device,
|
|
Discover,
|
|
UnsupportedDeviceError,
|
|
)
|
|
from kasa.discover import DiscoveryResult
|
|
|
|
from .common import echo
|
|
|
|
|
|
@click.command()
|
|
@click.pass_context
|
|
async def discover(ctx):
|
|
"""Discover devices in the network."""
|
|
target = ctx.parent.params["target"]
|
|
username = ctx.parent.params["username"]
|
|
password = ctx.parent.params["password"]
|
|
discovery_timeout = ctx.parent.params["discovery_timeout"]
|
|
timeout = ctx.parent.params["timeout"]
|
|
host = ctx.parent.params["host"]
|
|
port = ctx.parent.params["port"]
|
|
|
|
credentials = Credentials(username, password) if username and password else None
|
|
|
|
sem = asyncio.Semaphore()
|
|
discovered = dict()
|
|
unsupported = []
|
|
auth_failed = []
|
|
|
|
async def print_unsupported(unsupported_exception: UnsupportedDeviceError):
|
|
unsupported.append(unsupported_exception)
|
|
async with sem:
|
|
if unsupported_exception.discovery_result:
|
|
echo("== Unsupported device ==")
|
|
_echo_discovery_info(unsupported_exception.discovery_result)
|
|
echo()
|
|
else:
|
|
echo("== Unsupported device ==")
|
|
echo(f"\t{unsupported_exception}")
|
|
echo()
|
|
|
|
from .device import state
|
|
|
|
async def print_discovered(dev: Device):
|
|
async with sem:
|
|
try:
|
|
await dev.update()
|
|
except AuthenticationError:
|
|
auth_failed.append(dev._discovery_info)
|
|
echo("== Authentication failed for device ==")
|
|
_echo_discovery_info(dev._discovery_info)
|
|
echo()
|
|
else:
|
|
ctx.parent.obj = dev
|
|
await ctx.parent.invoke(state)
|
|
discovered[dev.host] = dev.internal_state
|
|
echo()
|
|
|
|
if host:
|
|
echo(f"Discovering device {host} for {discovery_timeout} seconds")
|
|
return await Discover.discover_single(
|
|
host,
|
|
port=port,
|
|
credentials=credentials,
|
|
timeout=timeout,
|
|
discovery_timeout=discovery_timeout,
|
|
on_unsupported=print_unsupported,
|
|
)
|
|
|
|
echo(f"Discovering devices on {target} for {discovery_timeout} seconds")
|
|
discovered_devices = await Discover.discover(
|
|
target=target,
|
|
discovery_timeout=discovery_timeout,
|
|
on_discovered=print_discovered,
|
|
on_unsupported=print_unsupported,
|
|
port=port,
|
|
timeout=timeout,
|
|
credentials=credentials,
|
|
)
|
|
|
|
for device in discovered_devices.values():
|
|
await device.protocol.close()
|
|
|
|
echo(f"Found {len(discovered)} devices")
|
|
if unsupported:
|
|
echo(f"Found {len(unsupported)} unsupported devices")
|
|
if auth_failed:
|
|
echo(f"Found {len(auth_failed)} devices that failed to authenticate")
|
|
|
|
return discovered
|
|
|
|
|
|
def _echo_dictionary(discovery_info: dict):
|
|
echo("\t[bold]== Discovery information ==[/bold]")
|
|
for key, value in discovery_info.items():
|
|
key_name = " ".join(x.capitalize() or "_" for x in key.split("_"))
|
|
key_name_and_spaces = "{:<15}".format(key_name + ":")
|
|
echo(f"\t{key_name_and_spaces}{value}")
|
|
|
|
|
|
def _echo_discovery_info(discovery_info):
|
|
# We don't have discovery info when all connection params are passed manually
|
|
if discovery_info is None:
|
|
return
|
|
|
|
if "system" in discovery_info and "get_sysinfo" in discovery_info["system"]:
|
|
_echo_dictionary(discovery_info["system"]["get_sysinfo"])
|
|
return
|
|
|
|
try:
|
|
dr = DiscoveryResult(**discovery_info)
|
|
except ValidationError:
|
|
_echo_dictionary(discovery_info)
|
|
return
|
|
|
|
def _conditional_echo(label, value):
|
|
if value:
|
|
ws = " " * (19 - len(label))
|
|
echo(f"\t{label}:{ws}{value}")
|
|
|
|
echo("\t[bold]== Discovery Result ==[/bold]")
|
|
_conditional_echo("Device Type", dr.device_type)
|
|
_conditional_echo("Device Model", dr.device_model)
|
|
_conditional_echo("Device Name", dr.device_name)
|
|
_conditional_echo("IP", dr.ip)
|
|
_conditional_echo("MAC", dr.mac)
|
|
_conditional_echo("Device Id (hash)", dr.device_id)
|
|
_conditional_echo("Owner (hash)", dr.owner)
|
|
_conditional_echo("FW Ver", dr.firmware_version)
|
|
_conditional_echo("HW Ver", dr.hw_ver)
|
|
_conditional_echo("HW Ver", dr.hardware_version)
|
|
_conditional_echo("Supports IOT Cloud", dr.is_support_iot_cloud)
|
|
_conditional_echo("OBD Src", dr.owner)
|
|
_conditional_echo("Factory Default", dr.factory_default)
|
|
_conditional_echo("Encrypt Type", dr.mgt_encrypt_schm.encrypt_type)
|
|
_conditional_echo("Encrypt Type", dr.encrypt_type)
|
|
_conditional_echo("Supports HTTPS", dr.mgt_encrypt_schm.is_support_https)
|
|
_conditional_echo("HTTP Port", dr.mgt_encrypt_schm.http_port)
|
|
_conditional_echo("Encrypt info", pf(dr.encrypt_info) if dr.encrypt_info else None)
|
|
_conditional_echo("Decrypted", pf(dr.decrypted_data) if dr.decrypted_data else None)
|
|
|
|
|
|
async def find_host_from_alias(alias, target="255.255.255.255", timeout=1, attempts=3):
|
|
"""Discover a device identified by its alias."""
|
|
for _attempt in range(1, attempts):
|
|
found_devs = await Discover.discover(target=target, timeout=timeout)
|
|
for _ip, dev in found_devs.items():
|
|
if dev.alias.lower() == alias.lower():
|
|
host = dev.host
|
|
return host
|
|
|
|
return None
|