mirror of
https://github.com/python-kasa/python-kasa.git
synced 2025-01-10 23:07:08 +00:00
8a131e1eeb
Instead of leveraging hound & travis, add an option for azure pipelines, which may replace the former in the future. This also streamlines the contributing guidelines by: * Adding pre-commit hooks to run isort, black, flake8, mypy * Adding lint environment to allow checks to be run `tox -e lint` This also contains a major cleanup to the SmartStrip handling which was due. After seeing #184 I thought I should push this in as it is for comments before the codebase diverges too much.
120 lines
3.6 KiB
Python
Executable File
120 lines
3.6 KiB
Python
Executable File
"""Implementation of the TP-Link Smart Home Protocol.
|
|
|
|
Encryption/Decryption methods based on the works of
|
|
Lubomir Stroetmann and Tobias Esser
|
|
|
|
https://www.softscheck.com/en/reverse-engineering-tp-link-hs110/
|
|
https://github.com/softScheck/tplink-smartplug/
|
|
|
|
which are licensed under the Apache License, Version 2.0
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
"""
|
|
import json
|
|
import logging
|
|
import socket
|
|
import struct
|
|
from typing import Any, Dict, Union
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
class TPLinkSmartHomeProtocol:
|
|
"""Implementation of the TP-Link Smart Home protocol."""
|
|
|
|
INITIALIZATION_VECTOR = 171
|
|
DEFAULT_PORT = 9999
|
|
DEFAULT_TIMEOUT = 5
|
|
|
|
@staticmethod
|
|
def query(host: str, request: Union[str, Dict], port: int = DEFAULT_PORT) -> Any:
|
|
"""Request information from a TP-Link SmartHome Device.
|
|
|
|
:param str host: host name or ip address of the device
|
|
:param int port: port on the device (default: 9999)
|
|
:param request: command to send to the device (can be either dict or
|
|
json string)
|
|
:return: response dict
|
|
"""
|
|
if isinstance(request, dict):
|
|
request = json.dumps(request)
|
|
|
|
timeout = TPLinkSmartHomeProtocol.DEFAULT_TIMEOUT
|
|
sock = None
|
|
try:
|
|
sock = socket.create_connection((host, port), timeout)
|
|
|
|
_LOGGER.debug("> (%i) %s", len(request), request)
|
|
sock.send(TPLinkSmartHomeProtocol.encrypt(request))
|
|
|
|
buffer = bytes()
|
|
# Some devices send responses with a length header of 0 and
|
|
# terminate with a zero size chunk. Others send the length and
|
|
# will hang if we attempt to read more data.
|
|
length = -1
|
|
while True:
|
|
chunk = sock.recv(4096)
|
|
if length == -1:
|
|
length = struct.unpack(">I", chunk[0:4])[0]
|
|
buffer += chunk
|
|
if (length > 0 and len(buffer) >= length + 4) or not chunk:
|
|
break
|
|
|
|
finally:
|
|
try:
|
|
if sock:
|
|
sock.shutdown(socket.SHUT_RDWR)
|
|
except OSError:
|
|
# OSX raises OSError when shutdown() gets called on a closed
|
|
# socket. We ignore it here as the data has already been read
|
|
# into the buffer at this point.
|
|
pass
|
|
|
|
finally:
|
|
if sock:
|
|
sock.close()
|
|
|
|
response = TPLinkSmartHomeProtocol.decrypt(buffer[4:])
|
|
_LOGGER.debug("< (%i) %s", len(response), response)
|
|
|
|
return json.loads(response)
|
|
|
|
@staticmethod
|
|
def encrypt(request: str) -> bytes:
|
|
"""
|
|
Encrypt a request for a TP-Link Smart Home Device.
|
|
|
|
:param request: plaintext request data
|
|
:return: ciphertext request
|
|
"""
|
|
key = TPLinkSmartHomeProtocol.INITIALIZATION_VECTOR
|
|
|
|
plainbytes = request.encode()
|
|
buffer = bytearray(struct.pack(">I", len(plainbytes)))
|
|
|
|
for plainbyte in plainbytes:
|
|
cipherbyte = key ^ plainbyte
|
|
key = cipherbyte
|
|
buffer.append(cipherbyte)
|
|
|
|
return bytes(buffer)
|
|
|
|
@staticmethod
|
|
def decrypt(ciphertext: bytes) -> str:
|
|
"""
|
|
Decrypt a response of a TP-Link Smart Home Device.
|
|
|
|
:param ciphertext: encrypted response data
|
|
:return: plaintext response
|
|
"""
|
|
key = TPLinkSmartHomeProtocol.INITIALIZATION_VECTOR
|
|
buffer = []
|
|
|
|
for cipherbyte in ciphertext:
|
|
plainbyte = key ^ cipherbyte
|
|
key = cipherbyte
|
|
buffer.append(plainbyte)
|
|
|
|
plaintext = bytes(buffer)
|
|
|
|
return plaintext.decode()
|