mirror of
https://github.com/python-kasa/python-kasa.git
synced 2025-01-08 22:07:06 +00:00
3490a1ef84
Add a tutorial module with examples that can be tested with `doctest`. In order to simplify the examples they can be run with doctest allowing top level await statements by adding a fixture to patch the builtins that xdoctest uses to test code. --------- Co-authored-by: Teemu R. <tpr@iki.fi>
203 lines
7.3 KiB
ReStructuredText
203 lines
7.3 KiB
ReStructuredText
.. py:module:: kasa.modules
|
|
|
|
|
|
.. _library_design:
|
|
|
|
Library Design & Modules
|
|
========================
|
|
|
|
This page aims to provide some details on the design and internals of this library.
|
|
You might be interested in this if you want to improve this library,
|
|
or if you are just looking to access some information that is not currently exposed.
|
|
|
|
.. contents:: Contents
|
|
:local:
|
|
|
|
.. _initialization:
|
|
|
|
Initialization
|
|
**************
|
|
|
|
Use :func:`~kasa.Discover.discover` to perform udp-based broadcast discovery on the network.
|
|
This will return you a list of device instances based on the discovery replies.
|
|
|
|
If the device's host is already known, you can use to construct a device instance with
|
|
:meth:`~kasa.Device.connect()`.
|
|
|
|
The :meth:`~kasa.Device.connect()` also enables support for connecting to new
|
|
KASA SMART protocol and TAPO devices directly using the parameter :class:`~kasa.DeviceConfig`.
|
|
Simply serialize the :attr:`~kasa.Device.config` property via :meth:`~kasa.DeviceConfig.to_dict()`
|
|
and then deserialize it later with :func:`~kasa.DeviceConfig.from_dict()`
|
|
and then pass it into :meth:`~kasa.Device.connect()`.
|
|
|
|
|
|
.. _update_cycle:
|
|
|
|
Update Cycle
|
|
************
|
|
|
|
When :meth:`~kasa.Device.update()` is called,
|
|
the library constructs a query to send to the device based on :ref:`supported modules <modules>`.
|
|
Internally, each module defines :meth:`~kasa.modules.Module.query()` to describe what they want query during the update.
|
|
|
|
The returned data is cached internally to avoid I/O on property accesses.
|
|
All properties defined both in the device class and in the module classes follow this principle.
|
|
|
|
While the properties are designed to provide a nice API to use for common use cases,
|
|
you may sometimes want to access the raw, cached data as returned by the device.
|
|
This can be done using the :attr:`~kasa.Device.internal_state` property.
|
|
|
|
|
|
.. _modules:
|
|
|
|
Modules
|
|
*******
|
|
|
|
The functionality provided by all :class:`~kasa.Device` instances is (mostly) done inside separate modules.
|
|
While the individual device-type specific classes provide an easy access for the most import features,
|
|
you can also access individual modules through :attr:`kasa.SmartDevice.modules`.
|
|
You can get the list of supported modules for a given device instance using :attr:`~kasa.Device.supported_modules`.
|
|
|
|
.. note::
|
|
|
|
If you only need some module-specific information,
|
|
you can call the wanted method on the module to avoid using :meth:`~kasa.Device.update`.
|
|
|
|
Protocols and Transports
|
|
************************
|
|
|
|
The library supports two different TP-Link protocols, ``IOT`` and ``SMART``.
|
|
``IOT`` is the original Kasa protocol and ``SMART`` is the newer protocol supported by TAPO devices and newer KASA devices.
|
|
The original protocol has a ``target``, ``command``, ``args`` interface whereas the new protocol uses a different set of
|
|
commands and has a ``method``, ``parameters`` interface.
|
|
Confusingly TP-Link originally called the Kasa line "Kasa Smart" and hence this library used "Smart" in a lot of the
|
|
module and class names but actually they were built to work with the ``IOT`` protocol.
|
|
|
|
In 2021 TP-Link started updating the underlying communication transport used by Kasa devices to make them more secure.
|
|
It switched from a TCP connection with static XOR type of encryption to a transport called ``KLAP`` which communicates
|
|
over http and uses handshakes to negotiate a dynamic encryption cipher.
|
|
This automatic update was put on hold and only seemed to affect UK HS100 models.
|
|
|
|
In 2023 TP-Link started updating the underlying communication transport used by Tapo devices to make them more secure.
|
|
It switched from AES encryption via public key exchange to use ``KLAP`` encryption and negotiation due to concerns
|
|
around impersonation with AES.
|
|
The encryption cipher is the same as for Kasa KLAP but the handshake seeds are slightly different.
|
|
Also in 2023 TP-Link started releasing newer Kasa branded devices using the ``SMART`` protocol.
|
|
This appears to be driven by hardware version rather than firmware.
|
|
|
|
|
|
In order to support these different configurations the library migrated from a single protocol class ``TPLinkSmartHomeProtocol``
|
|
to support pluggable transports and protocols.
|
|
The classes providing this functionality are:
|
|
|
|
- :class:`BaseProtocol <kasa.protocol.BaseProtocol>`
|
|
- :class:`IotProtocol <kasa.iotprotocol.IotProtocol>`
|
|
- :class:`SmartProtocol <kasa.smartprotocol.SmartProtocol>`
|
|
|
|
- :class:`BaseTransport <kasa.protocol.BaseTransport>`
|
|
- :class:`XorTransport <kasa.xortransport.XorTransport>`
|
|
- :class:`AesTransport <kasa.aestransport.AesTransport>`
|
|
- :class:`KlapTransport <kasa.klaptransport.KlapTransport>`
|
|
- :class:`KlapTransportV2 <kasa.klaptransport.KlapTransportV2>`
|
|
|
|
Errors and Exceptions
|
|
*********************
|
|
|
|
The base exception for all library errors is :class:`KasaException <kasa.exceptions.KasaException>`.
|
|
|
|
- If the device returns an error the library raises a :class:`DeviceError <kasa.exceptions.DeviceError>` which will usually contain an ``error_code`` with the detail.
|
|
- If the device fails to authenticate the library raises an :class:`AuthenticationError <kasa.exceptions.AuthenticationError>` which is derived
|
|
from :class:`DeviceError <kasa.exceptions.DeviceError>` and could contain an ``error_code`` depending on the type of failure.
|
|
- If the library encounters and unsupported deviceit raises an :class:`UnsupportedDeviceError <kasa.exceptions.UnsupportedDeviceError>`.
|
|
- If the device fails to respond within a timeout the library raises a :class:`TimeoutError <kasa.exceptions.TimeoutError>`.
|
|
- All other failures will raise the base :class:`KasaException <kasa.exceptions.KasaException>` class.
|
|
|
|
API documentation for modules and features
|
|
******************************************
|
|
|
|
.. autoclass:: kasa.Module
|
|
:noindex:
|
|
:members:
|
|
:inherited-members:
|
|
:undoc-members:
|
|
|
|
.. automodule:: kasa.interfaces
|
|
:noindex:
|
|
:members:
|
|
:inherited-members:
|
|
:undoc-members:
|
|
|
|
.. autoclass:: kasa.Feature
|
|
:noindex:
|
|
:members:
|
|
:inherited-members:
|
|
:undoc-members:
|
|
|
|
|
|
|
|
API documentation for protocols and transports
|
|
**********************************************
|
|
|
|
.. autoclass:: kasa.protocol.BaseProtocol
|
|
:members:
|
|
:inherited-members:
|
|
:undoc-members:
|
|
|
|
.. autoclass:: kasa.iotprotocol.IotProtocol
|
|
:members:
|
|
:inherited-members:
|
|
:undoc-members:
|
|
|
|
.. autoclass:: kasa.smartprotocol.SmartProtocol
|
|
:members:
|
|
:inherited-members:
|
|
:undoc-members:
|
|
|
|
.. autoclass:: kasa.protocol.BaseTransport
|
|
:members:
|
|
:inherited-members:
|
|
:undoc-members:
|
|
|
|
.. autoclass:: kasa.xortransport.XorTransport
|
|
:members:
|
|
:inherited-members:
|
|
:undoc-members:
|
|
|
|
.. autoclass:: kasa.klaptransport.KlapTransport
|
|
:members:
|
|
:inherited-members:
|
|
:undoc-members:
|
|
|
|
.. autoclass:: kasa.klaptransport.KlapTransportV2
|
|
:members:
|
|
:inherited-members:
|
|
:undoc-members:
|
|
|
|
.. autoclass:: kasa.aestransport.AesTransport
|
|
:members:
|
|
:inherited-members:
|
|
:undoc-members:
|
|
|
|
API documentation for errors and exceptions
|
|
*******************************************
|
|
|
|
.. autoclass:: kasa.exceptions.KasaException
|
|
:members:
|
|
:undoc-members:
|
|
|
|
.. autoclass:: kasa.exceptions.DeviceError
|
|
:members:
|
|
:undoc-members:
|
|
|
|
.. autoclass:: kasa.exceptions.AuthenticationError
|
|
:members:
|
|
:undoc-members:
|
|
|
|
.. autoclass:: kasa.exceptions.UnsupportedDeviceError
|
|
:members:
|
|
:undoc-members:
|
|
|
|
.. autoclass:: kasa.exceptions.TimeoutError
|
|
:members:
|
|
:undoc-members:
|