2023-11-11 00:14:00 +01:00
|
|
|
"""Test fixtures."""
|
|
|
|
from __future__ import annotations
|
|
|
|
|
|
|
|
import asyncio
|
|
|
|
import socket
|
|
|
|
from unittest.mock import MagicMock, patch
|
|
|
|
|
|
|
|
import pytest
|
|
|
|
import pytest_asyncio
|
|
|
|
|
|
|
|
from aioesphomeapi._frame_helper import APIPlaintextFrameHelper
|
|
|
|
from aioesphomeapi.client import APIClient, ConnectionParams
|
|
|
|
from aioesphomeapi.connection import APIConnection
|
|
|
|
from aioesphomeapi.host_resolver import AddrInfo, IPv4Sockaddr
|
2023-11-17 20:11:36 +01:00
|
|
|
from aioesphomeapi.zeroconf import ZeroconfManager
|
2023-11-11 00:14:00 +01:00
|
|
|
|
2023-11-17 20:11:36 +01:00
|
|
|
from .common import connect, get_mock_async_zeroconf, send_plaintext_hello
|
|
|
|
|
2023-11-21 14:01:58 +01:00
|
|
|
KEEP_ALIVE_INTERVAL = 15.0
|
|
|
|
|
2023-11-17 20:11:36 +01:00
|
|
|
|
|
|
|
@pytest.fixture
|
|
|
|
def async_zeroconf():
|
|
|
|
return get_mock_async_zeroconf()
|
2023-11-11 00:14:00 +01:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
|
|
|
def resolve_host():
|
|
|
|
with patch("aioesphomeapi.host_resolver.async_resolve_host") as func:
|
|
|
|
func.return_value = AddrInfo(
|
|
|
|
family=socket.AF_INET,
|
|
|
|
type=socket.SOCK_STREAM,
|
|
|
|
proto=socket.IPPROTO_TCP,
|
|
|
|
sockaddr=IPv4Sockaddr("10.0.0.512", 6052),
|
|
|
|
)
|
|
|
|
yield func
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
|
|
|
def socket_socket():
|
|
|
|
with patch("socket.socket") as func:
|
|
|
|
yield func
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
|
|
|
def connection_params() -> ConnectionParams:
|
|
|
|
return ConnectionParams(
|
|
|
|
address="fake.address",
|
|
|
|
port=6052,
|
|
|
|
password=None,
|
|
|
|
client_info="Tests client",
|
2023-11-21 14:01:58 +01:00
|
|
|
keepalive=KEEP_ALIVE_INTERVAL,
|
2023-11-17 20:11:36 +01:00
|
|
|
zeroconf_manager=ZeroconfManager(),
|
2023-11-11 00:14:00 +01:00
|
|
|
noise_psk=None,
|
|
|
|
expected_name=None,
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
2023-11-21 16:42:23 +01:00
|
|
|
def noise_connection_params() -> ConnectionParams:
|
|
|
|
return ConnectionParams(
|
|
|
|
address="fake.address",
|
|
|
|
port=6052,
|
|
|
|
password=None,
|
|
|
|
client_info="Tests client",
|
|
|
|
keepalive=KEEP_ALIVE_INTERVAL,
|
|
|
|
zeroconf_manager=ZeroconfManager(),
|
|
|
|
noise_psk="QRTIErOb/fcE9Ukd/5qA3RGYMn0Y+p06U58SCtOXvPc=",
|
|
|
|
expected_name="test",
|
|
|
|
)
|
2023-11-11 00:14:00 +01:00
|
|
|
|
2023-11-21 16:42:23 +01:00
|
|
|
|
|
|
|
async def on_stop(expected_disconnect: bool) -> None:
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
|
|
|
def conn(connection_params: ConnectionParams) -> APIConnection:
|
2023-11-11 00:14:00 +01:00
|
|
|
return APIConnection(connection_params, on_stop)
|
|
|
|
|
|
|
|
|
2023-11-21 16:42:23 +01:00
|
|
|
@pytest.fixture
|
|
|
|
def noise_conn(noise_connection_params: ConnectionParams) -> APIConnection:
|
|
|
|
return APIConnection(noise_connection_params, on_stop)
|
|
|
|
|
|
|
|
|
2023-11-11 00:14:00 +01:00
|
|
|
@pytest_asyncio.fixture(name="plaintext_connect_task_no_login")
|
|
|
|
async def plaintext_connect_task_no_login(
|
|
|
|
conn: APIConnection, resolve_host, socket_socket, event_loop
|
|
|
|
) -> tuple[APIConnection, asyncio.Transport, APIPlaintextFrameHelper, asyncio.Task]:
|
|
|
|
loop = asyncio.get_event_loop()
|
|
|
|
protocol: APIPlaintextFrameHelper | None = None
|
|
|
|
transport = MagicMock()
|
|
|
|
connected = asyncio.Event()
|
|
|
|
|
|
|
|
def _create_mock_transport_protocol(create_func, **kwargs):
|
|
|
|
nonlocal protocol
|
|
|
|
protocol = create_func()
|
|
|
|
protocol.connection_made(transport)
|
|
|
|
connected.set()
|
|
|
|
return transport, protocol
|
|
|
|
|
|
|
|
with patch.object(event_loop, "sock_connect"), patch.object(
|
|
|
|
loop, "create_connection", side_effect=_create_mock_transport_protocol
|
|
|
|
):
|
|
|
|
connect_task = asyncio.create_task(connect(conn, login=False))
|
|
|
|
await connected.wait()
|
|
|
|
yield conn, transport, protocol, connect_task
|
|
|
|
|
|
|
|
|
|
|
|
@pytest_asyncio.fixture(name="plaintext_connect_task_with_login")
|
|
|
|
async def plaintext_connect_task_with_login(
|
|
|
|
conn: APIConnection, resolve_host, socket_socket, event_loop
|
|
|
|
) -> tuple[APIConnection, asyncio.Transport, APIPlaintextFrameHelper, asyncio.Task]:
|
|
|
|
loop = asyncio.get_event_loop()
|
|
|
|
protocol: APIPlaintextFrameHelper | None = None
|
|
|
|
transport = MagicMock()
|
|
|
|
connected = asyncio.Event()
|
|
|
|
|
|
|
|
def _create_mock_transport_protocol(create_func, **kwargs):
|
|
|
|
nonlocal protocol
|
|
|
|
protocol = create_func()
|
|
|
|
protocol.connection_made(transport)
|
|
|
|
connected.set()
|
|
|
|
return transport, protocol
|
|
|
|
|
|
|
|
with patch.object(event_loop, "sock_connect"), patch.object(
|
|
|
|
loop, "create_connection", side_effect=_create_mock_transport_protocol
|
|
|
|
):
|
|
|
|
connect_task = asyncio.create_task(connect(conn, login=True))
|
|
|
|
await connected.wait()
|
|
|
|
yield conn, transport, protocol, connect_task
|
|
|
|
|
|
|
|
|
|
|
|
@pytest_asyncio.fixture(name="api_client")
|
|
|
|
async def api_client(
|
|
|
|
conn: APIConnection, resolve_host, socket_socket, event_loop
|
|
|
|
) -> tuple[APIClient, APIConnection, asyncio.Transport, APIPlaintextFrameHelper]:
|
|
|
|
loop = asyncio.get_event_loop()
|
|
|
|
protocol: APIPlaintextFrameHelper | None = None
|
|
|
|
transport = MagicMock()
|
|
|
|
connected = asyncio.Event()
|
|
|
|
client = APIClient(
|
|
|
|
address="mydevice.local",
|
|
|
|
port=6052,
|
|
|
|
password=None,
|
|
|
|
)
|
|
|
|
|
|
|
|
def _create_mock_transport_protocol(create_func, **kwargs):
|
|
|
|
nonlocal protocol
|
|
|
|
protocol = create_func()
|
|
|
|
protocol.connection_made(transport)
|
|
|
|
connected.set()
|
|
|
|
return transport, protocol
|
|
|
|
|
|
|
|
with patch.object(event_loop, "sock_connect"), patch.object(
|
|
|
|
loop, "create_connection", side_effect=_create_mock_transport_protocol
|
|
|
|
):
|
|
|
|
connect_task = asyncio.create_task(connect(conn, login=False))
|
|
|
|
await connected.wait()
|
2023-11-11 20:06:27 +01:00
|
|
|
send_plaintext_hello(protocol)
|
2023-11-11 00:14:00 +01:00
|
|
|
client._connection = conn
|
|
|
|
await connect_task
|
|
|
|
transport.reset_mock()
|
|
|
|
yield client, conn, transport, protocol
|