Skip to content

Commit 34c9659

Browse files
ikalchevakxbdracopjkundertIvan Kalchev
authored
V4.9.2 (#482)
* Late-import base36 and QR code libraries; remove SUPPORT_QR_CODE flag * Increase idle connection check interval to 300s (#475) This check was creating a lot of TimerHandles when the user had multiple bridges. We do not need to check very often as connections usually stay around for 24+hours * Implement zerocopy writes for the encrypted protocol (#476) * Implement zerocopy writes for the encrypted protocol With Python 3.12+ and later `transport.writelines` is implemented as [`sendmsg(..., IOV_MAX)`](python/cpython#91166) which allows us to avoid joining the bytes and sending them in one go. Older Python will effectively do the same thing we do now `b"".join(...)` * update tests * Revert "Late-import base36 and QR code libraries; remove SUPPORT_QR_CODE flag" (#477) * Avoid os.chmod failing on Windows if file non-existant (#471) * Avoid os.chmod failing on Windows if file non-existant * Update accessory_driver.py --------- Co-authored-by: Ivan Kalchev <[email protected]> * Fix mdns tests (#478) * Fix pylint complaints (#480) * Address remaining pylint complaints (#481) * Address remaining pylint complaints * Address remaining pylint complaints * v4.9.2 --------- Co-authored-by: Aarni Koskela <[email protected]> Co-authored-by: J. Nick Koston <[email protected]> Co-authored-by: Perry Kundert <[email protected]> Co-authored-by: Ivan Kalchev <[email protected]>
1 parent 5265b54 commit 34c9659

12 files changed

+65
-53
lines changed

CHANGELOG.md

+5
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,11 @@ Sections
1616
### Developers
1717
-->
1818

19+
## [4.9.2] - 2024-11-03
20+
21+
- Implement zerocopy writes for the encrypted protocol. [#476](https://github.com/ikalchev/HAP-python/pull/476)
22+
- Linter and test fixe.
23+
1924
## [4.9.1] - 2023-10-25
2025

2126
- Fix handling of explict close. [#467](https://github.com/ikalchev/HAP-python/pull/467)

pyhap/accessory.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -207,7 +207,7 @@ def xhm_uri(self) -> str:
207207
int(self.driver.state.pincode.replace(b"-", b""), 10) & 0x7FFFFFFF
208208
) # pincode
209209

210-
encoded_payload = base36.dumps(payload).upper()
210+
encoded_payload = base36.dumps(payload).upper() # pylint: disable=possibly-used-before-assignment
211211
encoded_payload = encoded_payload.rjust(9, "0")
212212

213213
return "X-HM://" + encoded_payload + self.driver.state.setup_id

pyhap/accessory_driver.py

+5-2
Original file line numberDiff line numberDiff line change
@@ -323,7 +323,7 @@ def start(self):
323323
and os.name != "nt"
324324
):
325325
logger.debug("Setting child watcher")
326-
watcher = asyncio.SafeChildWatcher()
326+
watcher = asyncio.SafeChildWatcher() # pylint: disable=deprecated-class
327327
watcher.attach_loop(self.loop)
328328
asyncio.set_child_watcher(watcher)
329329
else:
@@ -642,16 +642,19 @@ def persist(self):
642642
tmp_filename = None
643643
try:
644644
temp_dir = os.path.dirname(self.persist_file)
645+
logger.debug("Creating temp persist file in '%s'", temp_dir)
645646
with tempfile.NamedTemporaryFile(
646647
mode="w", dir=temp_dir, delete=False
647648
) as file_handle:
648649
tmp_filename = file_handle.name
650+
logger.debug("Created temp persist file '%s' named '%s'", file_handle, tmp_filename)
649651
self.encoder.persist(file_handle, self.state)
650652
if (
651653
os.name == "nt"
652654
): # Or `[WinError 5] Access Denied` will be raised on Windows
653655
os.chmod(tmp_filename, 0o644)
654-
os.chmod(self.persist_file, 0o644)
656+
if os.path.exists(self.persist_file):
657+
os.chmod(self.persist_file, 0o644)
655658
os.replace(tmp_filename, self.persist_file)
656659
except Exception: # pylint: disable=broad-except
657660
logger.exception("Failed to persist accessory state")

pyhap/const.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
"""This module contains constants used by other modules."""
22
MAJOR_VERSION = 4
33
MINOR_VERSION = 9
4-
PATCH_VERSION = 1
4+
PATCH_VERSION = 2
55
__short_version__ = f"{MAJOR_VERSION}.{MINOR_VERSION}"
66
__version__ = f"{__short_version__}.{PATCH_VERSION}"
77
REQUIRED_PYTHON_VER = (3, 7)

pyhap/hap_crypto.py

+3-6
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@
33
import logging
44
import struct
55
from struct import Struct
6-
from typing import List
6+
from typing import Iterable, List
77

88
from chacha20poly1305_reuseable import ChaCha20Poly1305Reusable as ChaCha20Poly1305
99
from cryptography.hazmat.backends import default_backend
@@ -112,7 +112,7 @@ def decrypt(self) -> bytes:
112112

113113
return result
114114

115-
def encrypt(self, data: bytes) -> bytes:
115+
def encrypt(self, data: bytes) -> Iterable[bytes]:
116116
"""Encrypt and send the return bytes."""
117117
result: List[bytes] = []
118118
offset = 0
@@ -127,7 +127,4 @@ def encrypt(self, data: bytes) -> bytes:
127127
offset += length
128128
self._out_count += 1
129129

130-
# Join the result once instead of concatenating each time
131-
# as this is much faster than generating an new immutable
132-
# byte string each time.
133-
return b"".join(result)
130+
return result

pyhap/hap_protocol.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -104,7 +104,7 @@ def write(self, data: bytes) -> None:
104104
self.handler.client_uuid,
105105
data,
106106
)
107-
self.transport.write(result)
107+
self.transport.writelines(result)
108108
else:
109109
logger.debug(
110110
"%s (%s): Send unencrypted: %s",

pyhap/hap_server.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,7 @@
1616

1717
logger = logging.getLogger(__name__)
1818

19-
IDLE_CONNECTION_CHECK_INTERVAL_SECONDS = 120
19+
IDLE_CONNECTION_CHECK_INTERVAL_SECONDS = 300
2020

2121

2222
class HAPServer:

pyproject.toml

+1
Original file line numberDiff line numberDiff line change
@@ -77,6 +77,7 @@ disable = [
7777
"too-many-return-statements",
7878
"too-many-statements",
7979
"too-many-boolean-expressions",
80+
"too-many-positional-arguments",
8081
"unused-argument",
8182
"wrong-import-order",
8283
"unused-argument",

tests/test_accessory_driver.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -961,7 +961,7 @@ def test_mdns_service_info(driver: AccessoryDriver):
961961
assert mdns_info.server == "Test-Accessory-000000.local."
962962
assert mdns_info.port == port
963963
assert mdns_info.addresses == [b"\xac\x00\x00\x01"]
964-
assert mdns_info.properties == {
964+
assert mdns_info.decoded_properties == {
965965
"md": "Test Accessory",
966966
"pv": "1.1",
967967
"id": "00:00:00:00:00:00",
@@ -990,7 +990,7 @@ def test_mdns_service_info_with_specified_server(driver: AccessoryDriver):
990990
assert mdns_info.server == "hap1.local."
991991
assert mdns_info.port == port
992992
assert mdns_info.addresses == [b"\xac\x00\x00\x01"]
993-
assert mdns_info.properties == {
993+
assert mdns_info.decoded_properties == {
994994
"md": "Test Accessory",
995995
"pv": "1.1",
996996
"id": "00:00:00:00:00:00",

tests/test_hap_crypto.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -15,7 +15,7 @@ def test_round_trip():
1515
crypto.OUT_CIPHER_INFO = crypto.IN_CIPHER_INFO
1616
crypto.reset(key)
1717

18-
encrypted = bytearray(crypto.encrypt(plaintext))
18+
encrypted = bytearray(b"".join(crypto.encrypt(plaintext)))
1919

2020
# Receive no data
2121
assert crypto.decrypt() == b""

tests/test_hap_protocol.py

+42-36
Original file line numberDiff line numberDiff line change
@@ -246,13 +246,13 @@ def test_get_accessories_with_crypto(driver):
246246
hap_proto.hap_crypto = MockHAPCrypto()
247247
hap_proto.handler.is_encrypted = True
248248

249-
with patch.object(hap_proto.transport, "write") as writer:
249+
with patch.object(hap_proto.transport, "writelines") as writelines:
250250
hap_proto.data_received(
251251
b"GET /accessories HTTP/1.1\r\nHost: Bridge\\032C77C47._hap._tcp.local\r\n\r\n" # pylint: disable=line-too-long
252252
)
253253

254254
hap_proto.close()
255-
assert b"accessories" in writer.call_args_list[0][0][0]
255+
assert b"accessories" in b"".join(writelines.call_args_list[0][0])
256256

257257

258258
def test_get_characteristics_with_crypto(driver):
@@ -273,7 +273,7 @@ def test_get_characteristics_with_crypto(driver):
273273
hap_proto.hap_crypto = MockHAPCrypto()
274274
hap_proto.handler.is_encrypted = True
275275

276-
with patch.object(hap_proto.transport, "write") as writer:
276+
with patch.object(hap_proto.transport, "writelines") as writelines:
277277
hap_proto.data_received(
278278
b"GET /characteristics?id=3762173001.7 HTTP/1.1\r\nHost: HASS\\032Bridge\\032YPHW\\032B223AD._hap._tcp.local\r\n\r\n" # pylint: disable=line-too-long
279279
)
@@ -282,13 +282,15 @@ def test_get_characteristics_with_crypto(driver):
282282
)
283283

284284
hap_proto.close()
285-
assert b"Content-Length:" in writer.call_args_list[0][0][0]
286-
assert b"Transfer-Encoding: chunked\r\n\r\n" not in writer.call_args_list[0][0][0]
287-
assert b"-70402" in writer.call_args_list[0][0][0]
285+
joined0 = b"".join(writelines.call_args_list[0][0])
286+
assert b"Content-Length:" in joined0
287+
assert b"Transfer-Encoding: chunked\r\n\r\n" not in joined0
288+
assert b"-70402" in joined0
288289

289-
assert b"Content-Length:" in writer.call_args_list[1][0][0]
290-
assert b"Transfer-Encoding: chunked\r\n\r\n" not in writer.call_args_list[1][0][0]
291-
assert b"TestAcc" in writer.call_args_list[1][0][0]
290+
joined1 = b"".join(writelines.call_args_list[1][0])
291+
assert b"Content-Length:" in joined1
292+
assert b"Transfer-Encoding: chunked\r\n\r\n" not in joined1
293+
assert b"TestAcc" in joined1
292294

293295

294296
def test_set_characteristics_with_crypto(driver):
@@ -309,13 +311,15 @@ def test_set_characteristics_with_crypto(driver):
309311
hap_proto.hap_crypto = MockHAPCrypto()
310312
hap_proto.handler.is_encrypted = True
311313

312-
with patch.object(hap_proto.transport, "write") as writer:
314+
with patch.object(hap_proto.transport, "writelines") as writelines:
313315
hap_proto.data_received(
314316
b'PUT /characteristics HTTP/1.1\r\nHost: HASS12\\032AD1C22._hap._tcp.local\r\nContent-Length: 49\r\nContent-Type: application/hap+json\r\n\r\n{"characteristics":[{"aid":1,"iid":9,"ev":true}]}' # pylint: disable=line-too-long
315317
)
316318

317319
hap_proto.close()
318-
assert writer.call_args_list[0][0][0] == b"HTTP/1.1 204 No Content\r\n\r\n"
320+
assert (
321+
b"".join(writelines.call_args_list[0][0]) == b"HTTP/1.1 204 No Content\r\n\r\n"
322+
)
319323

320324

321325
def test_crypto_failure_closes_connection(driver):
@@ -352,14 +356,14 @@ def test_empty_encrypted_data(driver):
352356

353357
hap_proto.hap_crypto = MockHAPCrypto()
354358
hap_proto.handler.is_encrypted = True
355-
with patch.object(hap_proto.transport, "write") as writer:
359+
with patch.object(hap_proto.transport, "writelines") as writelines:
356360
hap_proto.data_received(b"")
357361
hap_proto.data_received(
358362
b"GET /accessories HTTP/1.1\r\nHost: Bridge\\032C77C47._hap._tcp.local\r\n\r\n" # pylint: disable=line-too-long
359363
)
360364

361365
hap_proto.close()
362-
assert b"accessories" in writer.call_args_list[0][0][0]
366+
assert b"accessories" in b"".join(writelines.call_args_list[0][0])
363367

364368

365369
def test_http_11_keep_alive(driver):
@@ -434,13 +438,13 @@ def test_camera_snapshot_without_snapshot_support(driver):
434438
hap_proto.hap_crypto = MockHAPCrypto()
435439
hap_proto.handler.is_encrypted = True
436440

437-
with patch.object(hap_proto.transport, "write") as writer:
441+
with patch.object(hap_proto.transport, "writelines") as writelines:
438442
hap_proto.data_received(
439443
b'POST /resource HTTP/1.1\r\nHost: HASS\\032Bridge\\032BROZ\\0323BF435._hap._tcp.local\r\nContent-Length: 79\r\nContent-Type: application/hap+json\r\n\r\n{"image-height":360,"resource-type":"image","image-width":640,"aid":1411620844}' # pylint: disable=line-too-long
440444
)
441445

442446
hap_proto.close()
443-
assert b"-70402" in writer.call_args_list[0][0][0]
447+
assert b"-70402" in b"".join(writelines.call_args_list[0][0])
444448

445449

446450
@pytest.mark.asyncio
@@ -464,14 +468,14 @@ def _get_snapshot(*_):
464468
hap_proto.hap_crypto = MockHAPCrypto()
465469
hap_proto.handler.is_encrypted = True
466470

467-
with patch.object(hap_proto.transport, "write") as writer:
471+
with patch.object(hap_proto.transport, "writelines") as writelines:
468472
hap_proto.data_received(
469473
b'POST /resource HTTP/1.1\r\nHost: HASS\\032Bridge\\032BROZ\\0323BF435._hap._tcp.local\r\nContent-Length: 79\r\nContent-Type: application/hap+json\r\n\r\n{"image-height":360,"resource-type":"image","image-width":640,"aid":1411620844}' # pylint: disable=line-too-long
470474
)
471475
await hap_proto.response.task
472476
await asyncio.sleep(0)
473477

474-
assert b"fakesnap" in writer.call_args_list[0][0][0]
478+
assert b"fakesnap" in b"".join(writelines.call_args_list[0][0])
475479

476480
hap_proto.close()
477481

@@ -497,14 +501,14 @@ async def _async_get_snapshot(*_):
497501
hap_proto.hap_crypto = MockHAPCrypto()
498502
hap_proto.handler.is_encrypted = True
499503

500-
with patch.object(hap_proto.transport, "write") as writer:
504+
with patch.object(hap_proto.transport, "writelines") as writelines:
501505
hap_proto.data_received(
502506
b'POST /resource HTTP/1.1\r\nHost: HASS\\032Bridge\\032BROZ\\0323BF435._hap._tcp.local\r\nContent-Length: 79\r\nContent-Type: application/hap+json\r\n\r\n{"image-height":360,"resource-type":"image","image-width":640,"aid":1411620844}' # pylint: disable=line-too-long
503507
)
504508
await hap_proto.response.task
505509
await asyncio.sleep(0)
506510

507-
assert b"fakesnap" in writer.call_args_list[0][0][0]
511+
assert b"fakesnap" in b"".join(writelines.call_args_list[0][0])
508512

509513
hap_proto.close()
510514

@@ -532,14 +536,14 @@ async def _async_get_snapshot(*_):
532536
hap_proto.handler.is_encrypted = True
533537

534538
with patch.object(hap_handler, "RESPONSE_TIMEOUT", 0.1), patch.object(
535-
hap_proto.transport, "write"
536-
) as writer:
539+
hap_proto.transport, "writelines"
540+
) as writelines:
537541
hap_proto.data_received(
538542
b'POST /resource HTTP/1.1\r\nHost: HASS\\032Bridge\\032BROZ\\0323BF435._hap._tcp.local\r\nContent-Length: 79\r\nContent-Type: application/hap+json\r\n\r\n{"image-height":360,"resource-type":"image","image-width":640,"aid":1411620844}' # pylint: disable=line-too-long
539543
)
540544
await asyncio.sleep(0.3)
541545

542-
assert b"-70402" in writer.call_args_list[0][0][0]
546+
assert b"-70402" in b"".join(writelines.call_args_list[0][0])
543547

544548
hap_proto.close()
545549

@@ -564,7 +568,7 @@ def _make_response(*_):
564568
response.shared_key = b"newkey"
565569
return response
566570

567-
with patch.object(hap_proto.transport, "write"), patch.object(
571+
with patch.object(hap_proto.transport, "writelines"), patch.object(
568572
hap_proto.handler, "dispatch", _make_response
569573
):
570574
hap_proto.data_received(
@@ -635,7 +639,7 @@ async def _async_get_snapshot(*_):
635639
hap_proto.hap_crypto = MockHAPCrypto()
636640
hap_proto.handler.is_encrypted = True
637641

638-
with patch.object(hap_proto.transport, "write") as writer:
642+
with patch.object(hap_proto.transport, "writelines") as writelines:
639643
hap_proto.data_received(
640644
b'POST /resource HTTP/1.1\r\nHost: HASS\\032Bridge\\032BROZ\\0323BF435._hap._tcp.local\r\nContent-Length: 79\r\nContent-Type: application/hap+json\r\n\r\n{"image-height":360,"resource-type":"image","image-width":640,"aid":1411620844}' # pylint: disable=line-too-long
641645
)
@@ -645,7 +649,7 @@ async def _async_get_snapshot(*_):
645649
pass
646650
await asyncio.sleep(0)
647651

648-
assert b"-70402" in writer.call_args_list[0][0][0]
652+
assert b"-70402" in b"".join(writelines.call_args_list[0][0])
649653

650654
hap_proto.close()
651655

@@ -671,7 +675,7 @@ def _get_snapshot(*_):
671675
hap_proto.hap_crypto = MockHAPCrypto()
672676
hap_proto.handler.is_encrypted = True
673677

674-
with patch.object(hap_proto.transport, "write") as writer:
678+
with patch.object(hap_proto.transport, "writelines") as writelines:
675679
hap_proto.data_received(
676680
b'POST /resource HTTP/1.1\r\nHost: HASS\\032Bridge\\032BROZ\\0323BF435._hap._tcp.local\r\nContent-Length: 79\r\nContent-Type: application/hap+json\r\n\r\n{"image-height":360,"resource-type":"image","image-width":640,"aid":1411620844}' # pylint: disable=line-too-long
677681
)
@@ -681,7 +685,7 @@ def _get_snapshot(*_):
681685
pass
682686
await asyncio.sleep(0)
683687

684-
assert b"-70402" in writer.call_args_list[0][0][0]
688+
assert b"-70402" in b"".join(writelines.call_args_list[0][0])
685689

686690
hap_proto.close()
687691

@@ -702,14 +706,14 @@ async def test_camera_snapshot_missing_accessory(driver):
702706
hap_proto.hap_crypto = MockHAPCrypto()
703707
hap_proto.handler.is_encrypted = True
704708

705-
with patch.object(hap_proto.transport, "write") as writer:
709+
with patch.object(hap_proto.transport, "writelines") as writelines:
706710
hap_proto.data_received(
707711
b'POST /resource HTTP/1.1\r\nHost: HASS\\032Bridge\\032BROZ\\0323BF435._hap._tcp.local\r\nContent-Length: 79\r\nContent-Type: application/hap+json\r\n\r\n{"image-height":360,"resource-type":"image","image-width":640,"aid":1411620844}' # pylint: disable=line-too-long
708712
)
709713
await asyncio.sleep(0)
710714

711715
assert hap_proto.response is None
712-
assert b"-70402" in writer.call_args_list[0][0][0]
716+
assert b"-70402" in b"".join(writelines.call_args_list[0][0])
713717
hap_proto.close()
714718

715719

@@ -777,20 +781,22 @@ def test_explicit_close(driver: AccessoryDriver):
777781
hap_proto.handler.is_encrypted = True
778782
assert hap_proto.transport.is_closing() is False
779783

780-
with patch.object(hap_proto.transport, "write") as writer:
784+
with patch.object(hap_proto.transport, "writelines") as writelines:
781785
hap_proto.data_received(
782786
b"GET /characteristics?id=3762173001.7 HTTP/1.1\r\nHost: HASS\\032Bridge\\032YPHW\\032B223AD._hap._tcp.local\r\n\r\n" # pylint: disable=line-too-long
783787
)
784788
hap_proto.data_received(
785789
b"GET /characteristics?id=1.5 HTTP/1.1\r\nConnection: close\r\nHost: HASS\\032Bridge\\032YPHW\\032B223AD._hap._tcp.local\r\n\r\n" # pylint: disable=line-too-long
786790
)
787791

788-
assert b"Content-Length:" in writer.call_args_list[0][0][0]
789-
assert b"Transfer-Encoding: chunked\r\n\r\n" not in writer.call_args_list[0][0][0]
790-
assert b"-70402" in writer.call_args_list[0][0][0]
792+
join0 = b"".join(writelines.call_args_list[0][0])
793+
assert b"Content-Length:" in join0
794+
assert b"Transfer-Encoding: chunked\r\n\r\n" not in join0
795+
assert b"-70402" in join0
791796

792-
assert b"Content-Length:" in writer.call_args_list[1][0][0]
793-
assert b"Transfer-Encoding: chunked\r\n\r\n" not in writer.call_args_list[1][0][0]
794-
assert b"TestAcc" in writer.call_args_list[1][0][0]
797+
join1 = b"".join(writelines.call_args_list[1][0])
798+
assert b"Content-Length:" in join1
799+
assert b"Transfer-Encoding: chunked\r\n\r\n" not in join1
800+
assert b"TestAcc" in join1
795801

796802
assert hap_proto.transport.is_closing() is True

tox.ini

+2-2
Original file line numberDiff line numberDiff line change
@@ -61,8 +61,8 @@ deps =
6161
-r{toxinidir}/requirements_all.txt
6262
-r{toxinidir}/requirements_test.txt
6363
commands =
64-
pylint pyhap --disable=missing-docstring,empty-docstring,invalid-name,fixme --max-line-length=120
65-
pylint tests --disable=duplicate-code,missing-docstring,empty-docstring,invalid-name,fixme --max-line-length=120
64+
pylint pyhap --disable=missing-docstring,empty-docstring,invalid-name,fixme,too-many-positional-arguments --max-line-length=120
65+
pylint tests --disable=duplicate-code,missing-docstring,empty-docstring,invalid-name,fixme,too-many-positional-arguments --max-line-length=120
6666

6767

6868
[testenv:bandit]

0 commit comments

Comments
 (0)