2023-10-23 19:32:20 +02:00
|
|
|
from __future__ import annotations
|
|
|
|
|
|
|
|
import asyncio
|
|
|
|
import time
|
|
|
|
from datetime import datetime, timezone
|
|
|
|
from functools import partial
|
2023-11-17 20:11:36 +01:00
|
|
|
from unittest.mock import AsyncMock, MagicMock, patch
|
2023-11-09 00:26:51 +01:00
|
|
|
|
2023-11-11 20:06:27 +01:00
|
|
|
from google.protobuf import message
|
2023-11-09 00:26:51 +01:00
|
|
|
from zeroconf import Zeroconf
|
2023-11-11 20:43:31 +01:00
|
|
|
from zeroconf.asyncio import AsyncZeroconf
|
2023-10-23 19:32:20 +02:00
|
|
|
|
2023-11-11 20:06:27 +01:00
|
|
|
from aioesphomeapi._frame_helper import APIPlaintextFrameHelper
|
2023-11-11 00:14:00 +01:00
|
|
|
from aioesphomeapi._frame_helper.plain_text import _cached_varuint_to_bytes
|
2023-11-11 20:06:27 +01:00
|
|
|
from aioesphomeapi.api_pb2 import ConnectResponse, HelloResponse
|
2023-11-11 00:14:00 +01:00
|
|
|
from aioesphomeapi.connection import APIConnection
|
|
|
|
from aioesphomeapi.core import MESSAGE_TYPE_TO_PROTO
|
|
|
|
|
2023-10-23 19:32:20 +02:00
|
|
|
UTC = timezone.utc
|
|
|
|
_MONOTONIC_RESOLUTION = time.get_clock_info("monotonic").resolution
|
|
|
|
# We use a partial here since it is implemented in native code
|
|
|
|
# and avoids the global lookup of UTC
|
|
|
|
utcnow: partial[datetime] = partial(datetime.now, UTC)
|
|
|
|
utcnow.__doc__ = "Get now in UTC time."
|
|
|
|
|
2023-11-11 00:14:00 +01:00
|
|
|
PROTO_TO_MESSAGE_TYPE = {v: k for k, v in MESSAGE_TYPE_TO_PROTO.items()}
|
|
|
|
|
2023-10-23 19:32:20 +02:00
|
|
|
|
2023-11-09 00:26:51 +01:00
|
|
|
def get_mock_zeroconf() -> MagicMock:
|
2023-11-17 20:11:36 +01:00
|
|
|
with patch("zeroconf.Zeroconf.start"):
|
|
|
|
zc = Zeroconf()
|
|
|
|
zc.close = MagicMock()
|
|
|
|
return zc
|
2023-11-09 00:26:51 +01:00
|
|
|
|
|
|
|
|
2023-11-17 20:11:36 +01:00
|
|
|
def get_mock_async_zeroconf() -> AsyncZeroconf:
|
|
|
|
aiozc = AsyncZeroconf(zc=get_mock_zeroconf())
|
|
|
|
aiozc.async_close = AsyncMock()
|
|
|
|
return aiozc
|
2023-11-11 20:43:31 +01:00
|
|
|
|
|
|
|
|
2023-11-09 00:26:51 +01:00
|
|
|
class Estr(str):
|
|
|
|
"""A subclassed string."""
|
|
|
|
|
|
|
|
|
2023-11-17 20:11:36 +01:00
|
|
|
def generate_plaintext_packet(msg: message.Message) -> bytes:
|
|
|
|
type_ = PROTO_TO_MESSAGE_TYPE[msg.__class__]
|
|
|
|
bytes_ = msg.SerializeToString()
|
2023-11-11 00:14:00 +01:00
|
|
|
return (
|
|
|
|
b"\0"
|
2023-11-17 20:11:36 +01:00
|
|
|
+ _cached_varuint_to_bytes(len(bytes_))
|
2023-11-11 00:14:00 +01:00
|
|
|
+ _cached_varuint_to_bytes(type_)
|
2023-11-17 20:11:36 +01:00
|
|
|
+ bytes_
|
2023-11-11 00:14:00 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
|
2023-10-23 19:32:20 +02:00
|
|
|
def as_utc(dattim: datetime) -> datetime:
|
|
|
|
"""Return a datetime as UTC time."""
|
|
|
|
if dattim.tzinfo == UTC:
|
|
|
|
return dattim
|
|
|
|
return dattim.astimezone(UTC)
|
|
|
|
|
|
|
|
|
|
|
|
def async_fire_time_changed(
|
|
|
|
datetime_: datetime | None = None, fire_all: bool = False
|
|
|
|
) -> None:
|
|
|
|
"""Fire a time changed event at an exact microsecond.
|
|
|
|
|
|
|
|
Consider that it is not possible to actually achieve an exact
|
|
|
|
microsecond in production as the event loop is not precise enough.
|
|
|
|
If your code relies on this level of precision, consider a different
|
|
|
|
approach, as this is only for testing.
|
|
|
|
"""
|
|
|
|
loop = asyncio.get_running_loop()
|
|
|
|
if datetime_ is None:
|
|
|
|
utc_datetime = datetime.now(UTC)
|
|
|
|
else:
|
|
|
|
utc_datetime = as_utc(datetime_)
|
|
|
|
|
|
|
|
timestamp = utc_datetime.timestamp()
|
|
|
|
for task in list(loop._scheduled):
|
|
|
|
if not isinstance(task, asyncio.TimerHandle):
|
|
|
|
continue
|
|
|
|
if task.cancelled():
|
|
|
|
continue
|
|
|
|
|
|
|
|
mock_seconds_into_future = timestamp - time.time()
|
|
|
|
future_seconds = task.when() - (loop.time() + _MONOTONIC_RESOLUTION)
|
|
|
|
|
|
|
|
if fire_all or mock_seconds_into_future >= future_seconds:
|
|
|
|
task._run()
|
|
|
|
task.cancel()
|
2023-11-11 00:14:00 +01:00
|
|
|
|
|
|
|
|
|
|
|
async def connect(conn: APIConnection, login: bool = True):
|
|
|
|
"""Wrapper for connection logic to do both parts."""
|
|
|
|
await conn.start_connection()
|
|
|
|
await conn.finish_connection(login=login)
|
2023-11-11 20:06:27 +01:00
|
|
|
|
|
|
|
|
|
|
|
def send_plaintext_hello(protocol: APIPlaintextFrameHelper) -> None:
|
|
|
|
hello_response: message.Message = HelloResponse()
|
|
|
|
hello_response.api_version_major = 1
|
|
|
|
hello_response.api_version_minor = 9
|
|
|
|
hello_response.name = "fake"
|
2023-11-17 20:11:36 +01:00
|
|
|
protocol.data_received(generate_plaintext_packet(hello_response))
|
2023-11-11 20:06:27 +01:00
|
|
|
|
|
|
|
|
|
|
|
def send_plaintext_connect_response(
|
|
|
|
protocol: APIPlaintextFrameHelper, invalid_password: bool
|
|
|
|
) -> None:
|
|
|
|
connect_response: message.Message = ConnectResponse()
|
|
|
|
connect_response.invalid_password = invalid_password
|
2023-11-17 20:11:36 +01:00
|
|
|
protocol.data_received(generate_plaintext_packet(connect_response))
|