Redact sensitive info from debug logs (#1069)

Redacts sensitive data when debug logging device responses such as mac,
location and usernames
This commit is contained in:
Steven B. 2024-07-17 18:57:09 +01:00 committed by GitHub
parent c19389f236
commit c4f015a2fb
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
11 changed files with 300 additions and 39 deletions

View File

@ -87,7 +87,8 @@ import ipaddress
import logging import logging
import socket import socket
from collections.abc import Awaitable from collections.abc import Awaitable
from typing import Callable, Dict, Optional, Type, cast from pprint import pformat as pf
from typing import Any, Callable, Dict, Optional, Type, cast
# When support for cpython older than 3.11 is dropped # When support for cpython older than 3.11 is dropped
# async_timeout can be replaced with asyncio.timeout # async_timeout can be replaced with asyncio.timeout
@ -112,8 +113,10 @@ from kasa.exceptions import (
UnsupportedDeviceError, UnsupportedDeviceError,
) )
from kasa.iot.iotdevice import IotDevice from kasa.iot.iotdevice import IotDevice
from kasa.iotprotocol import REDACTORS as IOT_REDACTORS
from kasa.json import dumps as json_dumps from kasa.json import dumps as json_dumps
from kasa.json import loads as json_loads from kasa.json import loads as json_loads
from kasa.protocol import mask_mac, redact_data
from kasa.xortransport import XorEncryption from kasa.xortransport import XorEncryption
_LOGGER = logging.getLogger(__name__) _LOGGER = logging.getLogger(__name__)
@ -123,6 +126,12 @@ OnDiscoveredCallable = Callable[[Device], Awaitable[None]]
OnUnsupportedCallable = Callable[[UnsupportedDeviceError], Awaitable[None]] OnUnsupportedCallable = Callable[[UnsupportedDeviceError], Awaitable[None]]
DeviceDict = Dict[str, Device] DeviceDict = Dict[str, Device]
NEW_DISCOVERY_REDACTORS: dict[str, Callable[[Any], Any] | None] = {
"device_id": lambda x: "REDACTED_" + x[9::],
"owner": lambda x: "REDACTED_" + x[9::],
"mac": mask_mac,
}
class _DiscoverProtocol(asyncio.DatagramProtocol): class _DiscoverProtocol(asyncio.DatagramProtocol):
"""Implementation of the discovery protocol handler. """Implementation of the discovery protocol handler.
@ -293,6 +302,8 @@ class Discover:
DISCOVERY_PORT_2 = 20002 DISCOVERY_PORT_2 = 20002
DISCOVERY_QUERY_2 = binascii.unhexlify("020000010000000000000000463cb5d3") DISCOVERY_QUERY_2 = binascii.unhexlify("020000010000000000000000463cb5d3")
_redact_data = True
@staticmethod @staticmethod
async def discover( async def discover(
*, *,
@ -484,7 +495,9 @@ class Discover:
f"Unable to read response from device: {config.host}: {ex}" f"Unable to read response from device: {config.host}: {ex}"
) from ex ) from ex
_LOGGER.debug("[DISCOVERY] %s << %s", config.host, info) if _LOGGER.isEnabledFor(logging.DEBUG):
data = redact_data(info, IOT_REDACTORS) if Discover._redact_data else info
_LOGGER.debug("[DISCOVERY] %s << %s", config.host, pf(data))
device_class = cast(Type[IotDevice], Discover._get_device_class(info)) device_class = cast(Type[IotDevice], Discover._get_device_class(info))
device = device_class(config.host, config=config) device = device_class(config.host, config=config)
@ -504,6 +517,7 @@ class Discover:
config: DeviceConfig, config: DeviceConfig,
) -> Device: ) -> Device:
"""Get SmartDevice from the new 20002 response.""" """Get SmartDevice from the new 20002 response."""
debug_enabled = _LOGGER.isEnabledFor(logging.DEBUG)
try: try:
info = json_loads(data[16:]) info = json_loads(data[16:])
except Exception as ex: except Exception as ex:
@ -514,8 +528,16 @@ class Discover:
try: try:
discovery_result = DiscoveryResult(**info["result"]) discovery_result = DiscoveryResult(**info["result"])
except ValidationError as ex: except ValidationError as ex:
if debug_enabled:
data = (
redact_data(info, NEW_DISCOVERY_REDACTORS)
if Discover._redact_data
else info
)
_LOGGER.debug( _LOGGER.debug(
"Unable to parse discovery from device %s: %s", config.host, info "Unable to parse discovery from device %s: %s",
config.host,
pf(data),
) )
raise UnsupportedDeviceError( raise UnsupportedDeviceError(
f"Unable to parse discovery from device: {config.host}: {ex}" f"Unable to parse discovery from device: {config.host}: {ex}"
@ -551,7 +573,13 @@ class Discover:
discovery_result=discovery_result.get_dict(), discovery_result=discovery_result.get_dict(),
) )
_LOGGER.debug("[DISCOVERY] %s << %s", config.host, info) if debug_enabled:
data = (
redact_data(info, NEW_DISCOVERY_REDACTORS)
if Discover._redact_data
else info
)
_LOGGER.debug("[DISCOVERY] %s << %s", config.host, pf(data))
device = device_class(config.host, protocol=protocol) device = device_class(config.host, protocol=protocol)
di = discovery_result.get_dict() di = discovery_result.get_dict()

View File

@ -4,6 +4,8 @@ from __future__ import annotations
import asyncio import asyncio
import logging import logging
from pprint import pformat as pf
from typing import Any, Callable
from .deviceconfig import DeviceConfig from .deviceconfig import DeviceConfig
from .exceptions import ( from .exceptions import (
@ -14,11 +16,26 @@ from .exceptions import (
_RetryableError, _RetryableError,
) )
from .json import dumps as json_dumps from .json import dumps as json_dumps
from .protocol import BaseProtocol, BaseTransport from .protocol import BaseProtocol, BaseTransport, mask_mac, redact_data
from .xortransport import XorEncryption, XorTransport from .xortransport import XorEncryption, XorTransport
_LOGGER = logging.getLogger(__name__) _LOGGER = logging.getLogger(__name__)
REDACTORS: dict[str, Callable[[Any], Any] | None] = {
"latitude": lambda x: 0,
"longitude": lambda x: 0,
"latitude_i": lambda x: 0,
"longitude_i": lambda x: 0,
"deviceId": lambda x: "REDACTED_" + x[9::],
"id": lambda x: "REDACTED_" + x[9::],
"alias": lambda x: "#MASKED_NAME#" if x else "",
"mac": mask_mac,
"mic_mac": mask_mac,
"ssid": lambda x: "#MASKED_SSID#" if x else "",
"oemId": lambda x: "REDACTED_" + x[9::],
"username": lambda _: "user@example.com", # cnCloud
}
class IotProtocol(BaseProtocol): class IotProtocol(BaseProtocol):
"""Class for the legacy TPLink IOT KASA Protocol.""" """Class for the legacy TPLink IOT KASA Protocol."""
@ -34,6 +51,7 @@ class IotProtocol(BaseProtocol):
super().__init__(transport=transport) super().__init__(transport=transport)
self._query_lock = asyncio.Lock() self._query_lock = asyncio.Lock()
self._redact_data = True
async def query(self, request: str | dict, retry_count: int = 3) -> dict: async def query(self, request: str | dict, retry_count: int = 3) -> dict:
"""Query the device retrying for retry_count on failure.""" """Query the device retrying for retry_count on failure."""
@ -85,7 +103,24 @@ class IotProtocol(BaseProtocol):
raise KasaException("Query reached somehow to unreachable") raise KasaException("Query reached somehow to unreachable")
async def _execute_query(self, request: str, retry_count: int) -> dict: async def _execute_query(self, request: str, retry_count: int) -> dict:
return await self._transport.send(request) debug_enabled = _LOGGER.isEnabledFor(logging.DEBUG)
if debug_enabled:
_LOGGER.debug(
"%s >> %s",
self._host,
request,
)
resp = await self._transport.send(request)
if debug_enabled:
data = redact_data(resp, REDACTORS) if self._redact_data else resp
_LOGGER.debug(
"%s << %s",
self._host,
pf(data),
)
return resp
async def close(self) -> None: async def close(self) -> None:
"""Close the underlying transport.""" """Close the underlying transport."""

View File

@ -50,7 +50,6 @@ import logging
import secrets import secrets
import struct import struct
import time import time
from pprint import pformat as pf
from typing import Any, cast from typing import Any, cast
from cryptography.hazmat.primitives import padding from cryptography.hazmat.primitives import padding
@ -353,7 +352,7 @@ class KlapTransport(BaseTransport):
+ f"request with seq {seq}" + f"request with seq {seq}"
) )
else: else:
_LOGGER.debug("Query posted " + msg) _LOGGER.debug("Device %s query posted %s", self._host, msg)
# Check for mypy # Check for mypy
if self._encryption_session is not None: if self._encryption_session is not None:
@ -361,11 +360,7 @@ class KlapTransport(BaseTransport):
json_payload = json_loads(decrypted_response) json_payload = json_loads(decrypted_response)
_LOGGER.debug( _LOGGER.debug("Device %s query response received", self._host)
"%s << %s",
self._host,
_LOGGER.isEnabledFor(logging.DEBUG) and pf(json_payload),
)
return json_payload return json_payload

View File

@ -18,6 +18,7 @@ import hashlib
import logging import logging
import struct import struct
from abc import ABC, abstractmethod from abc import ABC, abstractmethod
from typing import Any, Callable, TypeVar, cast
# When support for cpython older than 3.11 is dropped # When support for cpython older than 3.11 is dropped
# async_timeout can be replaced with asyncio.timeout # async_timeout can be replaced with asyncio.timeout
@ -28,6 +29,46 @@ _LOGGER = logging.getLogger(__name__)
_NO_RETRY_ERRORS = {errno.EHOSTDOWN, errno.EHOSTUNREACH, errno.ECONNREFUSED} _NO_RETRY_ERRORS = {errno.EHOSTDOWN, errno.EHOSTUNREACH, errno.ECONNREFUSED}
_UNSIGNED_INT_NETWORK_ORDER = struct.Struct(">I") _UNSIGNED_INT_NETWORK_ORDER = struct.Struct(">I")
_T = TypeVar("_T")
def redact_data(data: _T, redactors: dict[str, Callable[[Any], Any] | None]) -> _T:
"""Redact sensitive data for logging."""
if not isinstance(data, (dict, list)):
return data
if isinstance(data, list):
return cast(_T, [redact_data(val, redactors) for val in data])
redacted = {**data}
for key, value in redacted.items():
if value is None:
continue
if isinstance(value, str) and not value:
continue
if key in redactors:
if redactor := redactors[key]:
try:
redacted[key] = redactor(value)
except: # noqa: E722
redacted[key] = "**REDACTEX**"
else:
redacted[key] = "**REDACTED**"
elif isinstance(value, dict):
redacted[key] = redact_data(value, redactors)
elif isinstance(value, list):
redacted[key] = [redact_data(item, redactors) for item in value]
return cast(_T, redacted)
def mask_mac(mac: str) -> str:
"""Return mac address with last two octects blanked."""
delim = ":" if ":" in mac else "-"
rest = delim.join(format(s, "02x") for s in bytes.fromhex("000000"))
return f"{mac[:8]}{delim}{rest}"
def md5(payload: bytes) -> bytes: def md5(payload: bytes) -> bytes:
"""Return the MD5 hash of the payload.""" """Return the MD5 hash of the payload."""

View File

@ -193,11 +193,9 @@ class SmartDevice(Device):
if not self._features: if not self._features:
await self._initialize_features() await self._initialize_features()
_LOGGER.debug( if _LOGGER.isEnabledFor(logging.DEBUG):
"Update completed %s: %s", updated = self._last_update if first_update else resp
self.host, _LOGGER.debug("Update completed %s: %s", self.host, list(updated.keys()))
self._last_update if first_update else resp,
)
def _handle_module_post_update_hook(self, module: SmartModule) -> bool: def _handle_module_post_update_hook(self, module: SmartModule) -> bool:
try: try:

View File

@ -12,7 +12,7 @@ import logging
import time import time
import uuid import uuid
from pprint import pformat as pf from pprint import pformat as pf
from typing import Any from typing import Any, Callable
from .exceptions import ( from .exceptions import (
SMART_AUTHENTICATION_ERRORS, SMART_AUTHENTICATION_ERRORS,
@ -26,10 +26,31 @@ from .exceptions import (
_RetryableError, _RetryableError,
) )
from .json import dumps as json_dumps from .json import dumps as json_dumps
from .protocol import BaseProtocol, BaseTransport, md5 from .protocol import BaseProtocol, BaseTransport, mask_mac, md5, redact_data
_LOGGER = logging.getLogger(__name__) _LOGGER = logging.getLogger(__name__)
REDACTORS: dict[str, Callable[[Any], Any] | None] = {
"latitude": lambda x: 0,
"longitude": lambda x: 0,
"la": lambda x: 0, # lat on ks240
"lo": lambda x: 0, # lon on ks240
"device_id": lambda x: "REDACTED_" + x[9::],
"parent_device_id": lambda x: "REDACTED_" + x[9::], # Hub attached children
"original_device_id": lambda x: "REDACTED_" + x[9::], # Strip children
"nickname": lambda x: "I01BU0tFRF9OQU1FIw==" if x else "",
"mac": mask_mac,
"ssid": lambda x: "I01BU0tFRF9TU0lEIw=" if x else "",
"bssid": lambda _: "000000000000",
"oem_id": lambda x: "REDACTED_" + x[9::],
"setup_code": None, # matter
"setup_payload": None, # matter
"mfi_setup_code": None, # mfi_ for homekit
"mfi_setup_id": None,
"mfi_token_token": None,
"mfi_token_uuid": None,
}
class SmartProtocol(BaseProtocol): class SmartProtocol(BaseProtocol):
"""Class for the new TPLink SMART protocol.""" """Class for the new TPLink SMART protocol."""
@ -50,6 +71,7 @@ class SmartProtocol(BaseProtocol):
self._multi_request_batch_size = ( self._multi_request_batch_size = (
self._transport._config.batch_size or self.DEFAULT_MULTI_REQUEST_BATCH_SIZE self._transport._config.batch_size or self.DEFAULT_MULTI_REQUEST_BATCH_SIZE
) )
self._redact_data = True
def get_smart_request(self, method, params=None) -> str: def get_smart_request(self, method, params=None) -> str:
"""Get a request message as a string.""" """Get a request message as a string."""
@ -167,11 +189,15 @@ class SmartProtocol(BaseProtocol):
) )
response_step = await self._transport.send(smart_request) response_step = await self._transport.send(smart_request)
if debug_enabled: if debug_enabled:
if self._redact_data:
data = redact_data(response_step, REDACTORS)
else:
data = response_step
_LOGGER.debug( _LOGGER.debug(
"%s %s << %s", "%s %s << %s",
self._host, self._host,
batch_name, batch_name,
pf(response_step), pf(data),
) )
try: try:
self._handle_response_error_code(response_step, batch_name) self._handle_response_error_code(response_step, batch_name)

View File

@ -90,21 +90,26 @@ def create_discovery_mock(ip: str, fixture_data: dict):
query_data: dict query_data: dict
device_type: str device_type: str
encrypt_type: str encrypt_type: str
_datagram: bytes
login_version: int | None = None login_version: int | None = None
port_override: int | None = None port_override: int | None = None
@property
def _datagram(self) -> bytes:
if self.default_port == 9999:
return XorEncryption.encrypt(json_dumps(self.discovery_data))[4:]
else:
return (
b"\x02\x00\x00\x01\x01[\x00\x00\x00\x00\x00\x00W\xcev\xf8"
+ json_dumps(self.discovery_data).encode()
)
if "discovery_result" in fixture_data: if "discovery_result" in fixture_data:
discovery_data = {"result": fixture_data["discovery_result"]} discovery_data = {"result": fixture_data["discovery_result"].copy()}
device_type = fixture_data["discovery_result"]["device_type"] device_type = fixture_data["discovery_result"]["device_type"]
encrypt_type = fixture_data["discovery_result"]["mgt_encrypt_schm"][ encrypt_type = fixture_data["discovery_result"]["mgt_encrypt_schm"][
"encrypt_type" "encrypt_type"
] ]
login_version = fixture_data["discovery_result"]["mgt_encrypt_schm"].get("lv") login_version = fixture_data["discovery_result"]["mgt_encrypt_schm"].get("lv")
datagram = (
b"\x02\x00\x00\x01\x01[\x00\x00\x00\x00\x00\x00W\xcev\xf8"
+ json_dumps(discovery_data).encode()
)
dm = _DiscoveryMock( dm = _DiscoveryMock(
ip, ip,
80, 80,
@ -113,16 +118,14 @@ def create_discovery_mock(ip: str, fixture_data: dict):
fixture_data, fixture_data,
device_type, device_type,
encrypt_type, encrypt_type,
datagram,
login_version, login_version,
) )
else: else:
sys_info = fixture_data["system"]["get_sysinfo"] sys_info = fixture_data["system"]["get_sysinfo"]
discovery_data = {"system": {"get_sysinfo": sys_info}} discovery_data = {"system": {"get_sysinfo": sys_info.copy()}}
device_type = sys_info.get("mic_type") or sys_info.get("type") device_type = sys_info.get("mic_type") or sys_info.get("type")
encrypt_type = "XOR" encrypt_type = "XOR"
login_version = None login_version = None
datagram = XorEncryption.encrypt(json_dumps(discovery_data))[4:]
dm = _DiscoveryMock( dm = _DiscoveryMock(
ip, ip,
9999, 9999,
@ -131,7 +134,6 @@ def create_discovery_mock(ip: str, fixture_data: dict):
fixture_data, fixture_data,
device_type, device_type,
encrypt_type, encrypt_type,
datagram,
login_version, login_version,
) )

View File

@ -2,6 +2,7 @@
# ruff: noqa: S106 # ruff: noqa: S106
import asyncio import asyncio
import logging
import re import re
import socket import socket
from unittest.mock import MagicMock from unittest.mock import MagicMock
@ -565,3 +566,38 @@ async def test_do_discover_external_cancel(mocker):
with pytest.raises(asyncio.TimeoutError): with pytest.raises(asyncio.TimeoutError):
async with asyncio_timeout(0): async with asyncio_timeout(0):
await dp.wait_for_discovery_to_complete() await dp.wait_for_discovery_to_complete()
async def test_discovery_redaction(discovery_mock, caplog: pytest.LogCaptureFixture):
"""Test query sensitive info redaction."""
mac = "12:34:56:78:9A:BC"
if discovery_mock.default_port == 9999:
sysinfo = discovery_mock.discovery_data["system"]["get_sysinfo"]
if "mac" in sysinfo:
sysinfo["mac"] = mac
elif "mic_mac" in sysinfo:
sysinfo["mic_mac"] = mac
else:
discovery_mock.discovery_data["result"]["mac"] = mac
# Info no message logging
caplog.set_level(logging.INFO)
await Discover.discover()
assert mac not in caplog.text
caplog.set_level(logging.DEBUG)
# Debug no redaction
caplog.clear()
Discover._redact_data = False
await Discover.discover()
assert mac in caplog.text
# Debug redaction
caplog.clear()
Discover._redact_data = True
await Discover.discover()
assert mac not in caplog.text
assert "12:34:56:00:00:00" in caplog.text

View File

@ -8,9 +8,12 @@ import os
import pkgutil import pkgutil
import struct import struct
import sys import sys
from typing import cast
import pytest import pytest
from kasa.iot import IotDevice
from ..aestransport import AesTransport from ..aestransport import AesTransport
from ..credentials import Credentials from ..credentials import Credentials
from ..device import Device from ..device import Device
@ -21,8 +24,12 @@ from ..klaptransport import KlapTransport, KlapTransportV2
from ..protocol import ( from ..protocol import (
BaseProtocol, BaseProtocol,
BaseTransport, BaseTransport,
mask_mac,
redact_data,
) )
from ..xortransport import XorEncryption, XorTransport from ..xortransport import XorEncryption, XorTransport
from .conftest import device_iot
from .fakeprotocol_iot import FakeIotTransport
@pytest.mark.parametrize( @pytest.mark.parametrize(
@ -676,3 +683,63 @@ def test_deprecated_protocol():
host = "127.0.0.1" host = "127.0.0.1"
proto = TPLinkSmartHomeProtocol(host=host) proto = TPLinkSmartHomeProtocol(host=host)
assert proto.config.host == host assert proto.config.host == host
@device_iot
async def test_iot_queries_redaction(dev: IotDevice, caplog: pytest.LogCaptureFixture):
"""Test query sensitive info redaction."""
device_id = "123456789ABCDEF"
cast(FakeIotTransport, dev.protocol._transport).proto["system"]["get_sysinfo"][
"deviceId"
] = device_id
# Info no message logging
caplog.set_level(logging.INFO)
await dev.update()
assert device_id not in caplog.text
caplog.set_level(logging.DEBUG, logger="kasa")
# The fake iot protocol also logs so disable it
test_logger = logging.getLogger("kasa.tests.fakeprotocol_iot")
test_logger.setLevel(logging.INFO)
# Debug no redaction
caplog.clear()
cast(IotProtocol, dev.protocol)._redact_data = False
await dev.update()
assert device_id in caplog.text
# Debug redaction
caplog.clear()
cast(IotProtocol, dev.protocol)._redact_data = True
await dev.update()
assert device_id not in caplog.text
assert "REDACTED_" + device_id[9::] in caplog.text
async def test_redact_data():
"""Test redact data function."""
data = {
"device_id": "123456789ABCDEF",
"owner": "0987654",
"mac": "12:34:56:78:90:AB",
"ip": "192.168.1",
"no_val": None,
}
excpected_data = {
"device_id": "REDACTED_ABCDEF",
"owner": "**REDACTED**",
"mac": "12:34:56:00:00:00",
"ip": "**REDACTEX**",
"no_val": None,
}
REDACTORS = {
"device_id": lambda x: "REDACTED_" + x[9::],
"owner": None,
"mac": mask_mac,
"ip": lambda x: "127.0.0." + x.split(".")[3],
}
redacted_data = redact_data(data, REDACTORS)
assert redacted_data == excpected_data

View File

@ -1,8 +1,11 @@
import logging import logging
from typing import cast
import pytest import pytest
import pytest_mock import pytest_mock
from kasa.smart import SmartDevice
from ..exceptions import ( from ..exceptions import (
SMART_RETRYABLE_ERRORS, SMART_RETRYABLE_ERRORS,
DeviceError, DeviceError,
@ -10,6 +13,7 @@ from ..exceptions import (
SmartErrorCode, SmartErrorCode,
) )
from ..smartprotocol import SmartProtocol, _ChildProtocolWrapper from ..smartprotocol import SmartProtocol, _ChildProtocolWrapper
from .conftest import device_smart
from .fakeprotocol_smart import FakeSmartTransport from .fakeprotocol_smart import FakeSmartTransport
DUMMY_QUERY = {"foobar": {"foo": "bar", "bar": "foo"}} DUMMY_QUERY = {"foobar": {"foo": "bar", "bar": "foo"}}
@ -409,3 +413,34 @@ async def test_incomplete_list(mocker, caplog):
"Device 127.0.0.123 returned empty results list for method get_preset_rules" "Device 127.0.0.123 returned empty results list for method get_preset_rules"
in caplog.text in caplog.text
) )
@device_smart
async def test_smart_queries_redaction(
dev: SmartDevice, caplog: pytest.LogCaptureFixture
):
"""Test query sensitive info redaction."""
device_id = "123456789ABCDEF"
cast(FakeSmartTransport, dev.protocol._transport).info["get_device_info"][
"device_id"
] = device_id
# Info no message logging
caplog.set_level(logging.INFO)
await dev.update()
assert device_id not in caplog.text
caplog.set_level(logging.DEBUG)
# Debug no redaction
caplog.clear()
dev.protocol._redact_data = False
await dev.update()
assert device_id in caplog.text
# Debug redaction
caplog.clear()
dev.protocol._redact_data = True
await dev.update()
assert device_id not in caplog.text
assert "REDACTED_" + device_id[9::] in caplog.text

View File

@ -19,7 +19,6 @@ import logging
import socket import socket
import struct import struct
from collections.abc import Generator from collections.abc import Generator
from pprint import pformat as pf
# When support for cpython older than 3.11 is dropped # When support for cpython older than 3.11 is dropped
# async_timeout can be replaced with asyncio.timeout # async_timeout can be replaced with asyncio.timeout
@ -78,9 +77,8 @@ class XorTransport(BaseTransport):
"""Execute a query on the device and wait for the response.""" """Execute a query on the device and wait for the response."""
assert self.writer is not None # noqa: S101 assert self.writer is not None # noqa: S101
assert self.reader is not None # noqa: S101 assert self.reader is not None # noqa: S101
debug_log = _LOGGER.isEnabledFor(logging.DEBUG) _LOGGER.debug("Device %s sending query %s", self._host, request)
if debug_log:
_LOGGER.debug("%s >> %s", self._host, request)
self.writer.write(XorEncryption.encrypt(request)) self.writer.write(XorEncryption.encrypt(request))
await self.writer.drain() await self.writer.drain()
@ -90,8 +88,8 @@ class XorTransport(BaseTransport):
buffer = await self.reader.readexactly(length) buffer = await self.reader.readexactly(length)
response = XorEncryption.decrypt(buffer) response = XorEncryption.decrypt(buffer)
json_payload = json_loads(response) json_payload = json_loads(response)
if debug_log:
_LOGGER.debug("%s << %s", self._host, pf(json_payload)) _LOGGER.debug("Device %s query response received", self._host)
return json_payload return json_payload