2016-11-28 00:26:46 +00:00
|
|
|
"""Test the aiohttp client helper."""
|
2017-01-30 00:15:40 +00:00
|
|
|
import asyncio
|
2021-01-01 21:31:56 +00:00
|
|
|
from unittest.mock import Mock, patch
|
2016-11-28 00:26:46 +00:00
|
|
|
|
|
|
|
import aiohttp
|
2018-03-05 21:28:41 +00:00
|
|
|
import pytest
|
2016-11-28 00:26:46 +00:00
|
|
|
|
2022-04-04 15:57:48 +00:00
|
|
|
from homeassistant.components.mjpeg.const import (
|
|
|
|
CONF_MJPEG_URL,
|
|
|
|
CONF_STILL_IMAGE_URL,
|
|
|
|
DOMAIN as MJPEG_DOMAIN,
|
|
|
|
)
|
|
|
|
from homeassistant.const import (
|
|
|
|
CONF_AUTHENTICATION,
|
|
|
|
CONF_PASSWORD,
|
|
|
|
CONF_USERNAME,
|
|
|
|
CONF_VERIFY_SSL,
|
|
|
|
HTTP_BASIC_AUTHENTICATION,
|
|
|
|
)
|
2023-02-08 07:51:43 +00:00
|
|
|
from homeassistant.core import EVENT_HOMEASSISTANT_CLOSE, HomeAssistant
|
2016-11-28 00:26:46 +00:00
|
|
|
import homeassistant.helpers.aiohttp_client as client
|
2022-07-11 21:46:55 +00:00
|
|
|
from homeassistant.util.color import RGBColor
|
2022-04-04 15:57:48 +00:00
|
|
|
|
2023-10-05 17:52:26 +00:00
|
|
|
from tests.common import MockConfigEntry, MockModule, mock_integration
|
2023-02-20 10:42:56 +00:00
|
|
|
from tests.test_util.aiohttp import AiohttpClientMocker
|
2016-11-28 00:26:46 +00:00
|
|
|
|
2020-05-13 07:58:33 +00:00
|
|
|
|
|
|
|
@pytest.fixture(name="camera_client")
|
|
|
|
def camera_client_fixture(hass, hass_client):
|
2018-03-05 21:28:41 +00:00
|
|
|
"""Fixture to fetch camera streams."""
|
2022-04-04 15:57:48 +00:00
|
|
|
mock_config_entry = MockConfigEntry(
|
|
|
|
title="MJPEG Camera",
|
|
|
|
domain=MJPEG_DOMAIN,
|
|
|
|
options={
|
|
|
|
CONF_AUTHENTICATION: HTTP_BASIC_AUTHENTICATION,
|
|
|
|
CONF_MJPEG_URL: "http://example.com/mjpeg_stream",
|
|
|
|
CONF_PASSWORD: None,
|
|
|
|
CONF_STILL_IMAGE_URL: None,
|
|
|
|
CONF_USERNAME: None,
|
|
|
|
CONF_VERIFY_SSL: True,
|
|
|
|
},
|
|
|
|
)
|
|
|
|
mock_config_entry.add_to_hass(hass)
|
|
|
|
hass.loop.run_until_complete(
|
|
|
|
hass.config_entries.async_setup(mock_config_entry.entry_id)
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2020-06-02 15:54:13 +00:00
|
|
|
hass.loop.run_until_complete(hass.async_block_till_done())
|
2018-03-05 21:28:41 +00:00
|
|
|
|
2023-01-27 12:57:06 +00:00
|
|
|
return hass.loop.run_until_complete(hass_client())
|
2018-03-05 21:28:41 +00:00
|
|
|
|
|
|
|
|
2023-02-08 07:51:43 +00:00
|
|
|
async def test_get_clientsession_with_ssl(hass: HomeAssistant) -> None:
|
2020-04-07 16:33:23 +00:00
|
|
|
"""Test init clientsession with ssl."""
|
|
|
|
client.async_get_clientsession(hass)
|
2023-10-24 23:40:39 +00:00
|
|
|
verify_ssl = True
|
|
|
|
family = 0
|
2016-11-28 00:26:46 +00:00
|
|
|
|
2023-10-24 23:40:39 +00:00
|
|
|
client_session = hass.data[client.DATA_CLIENTSESSION][(verify_ssl, family)]
|
|
|
|
assert isinstance(client_session, aiohttp.ClientSession)
|
|
|
|
connector = hass.data[client.DATA_CONNECTOR][(verify_ssl, family)]
|
|
|
|
assert isinstance(connector, aiohttp.TCPConnector)
|
2016-11-28 00:26:46 +00:00
|
|
|
|
|
|
|
|
2023-02-08 07:51:43 +00:00
|
|
|
async def test_get_clientsession_without_ssl(hass: HomeAssistant) -> None:
|
2020-04-07 16:33:23 +00:00
|
|
|
"""Test init clientsession without ssl."""
|
|
|
|
client.async_get_clientsession(hass, verify_ssl=False)
|
2023-10-24 23:40:39 +00:00
|
|
|
verify_ssl = False
|
|
|
|
family = 0
|
2019-07-31 19:25:30 +00:00
|
|
|
|
2023-10-24 23:40:39 +00:00
|
|
|
client_session = hass.data[client.DATA_CLIENTSESSION][(verify_ssl, family)]
|
|
|
|
assert isinstance(client_session, aiohttp.ClientSession)
|
|
|
|
connector = hass.data[client.DATA_CONNECTOR][(verify_ssl, family)]
|
|
|
|
assert isinstance(connector, aiohttp.TCPConnector)
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
|
|
("verify_ssl", "expected_family"),
|
|
|
|
[(True, 0), (False, 0), (True, 4), (False, 4), (True, 6), (False, 6)],
|
|
|
|
)
|
|
|
|
async def test_get_clientsession(
|
|
|
|
hass: HomeAssistant, verify_ssl: bool, expected_family: int
|
|
|
|
) -> None:
|
|
|
|
"""Test init clientsession combinations."""
|
|
|
|
client.async_get_clientsession(hass, verify_ssl=verify_ssl, family=expected_family)
|
|
|
|
client_session = hass.data[client.DATA_CLIENTSESSION][(verify_ssl, expected_family)]
|
|
|
|
assert isinstance(client_session, aiohttp.ClientSession)
|
|
|
|
connector = hass.data[client.DATA_CONNECTOR][(verify_ssl, expected_family)]
|
|
|
|
assert isinstance(connector, aiohttp.TCPConnector)
|
2016-11-28 00:26:46 +00:00
|
|
|
|
|
|
|
|
2023-02-08 07:51:43 +00:00
|
|
|
async def test_create_clientsession_with_ssl_and_cookies(hass: HomeAssistant) -> None:
|
2020-04-07 16:33:23 +00:00
|
|
|
"""Test create clientsession with ssl."""
|
|
|
|
session = client.async_create_clientsession(hass, cookies={"bla": True})
|
|
|
|
assert isinstance(session, aiohttp.ClientSession)
|
2023-10-24 23:40:39 +00:00
|
|
|
|
|
|
|
verify_ssl = True
|
|
|
|
family = 0
|
|
|
|
|
|
|
|
assert client.DATA_CLIENTSESSION not in hass.data
|
|
|
|
connector = hass.data[client.DATA_CONNECTOR][(verify_ssl, family)]
|
|
|
|
assert isinstance(connector, aiohttp.TCPConnector)
|
2016-11-28 00:26:46 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
|
2023-02-08 07:51:43 +00:00
|
|
|
async def test_create_clientsession_without_ssl_and_cookies(
|
|
|
|
hass: HomeAssistant,
|
|
|
|
) -> None:
|
2020-04-07 16:33:23 +00:00
|
|
|
"""Test create clientsession without ssl."""
|
|
|
|
session = client.async_create_clientsession(hass, False, cookies={"bla": True})
|
|
|
|
assert isinstance(session, aiohttp.ClientSession)
|
2016-11-28 00:26:46 +00:00
|
|
|
|
2023-10-24 23:40:39 +00:00
|
|
|
verify_ssl = False
|
|
|
|
family = 0
|
2016-11-28 00:26:46 +00:00
|
|
|
|
2023-10-24 23:40:39 +00:00
|
|
|
assert client.DATA_CLIENTSESSION not in hass.data
|
|
|
|
connector = hass.data[client.DATA_CONNECTOR][(verify_ssl, family)]
|
|
|
|
assert isinstance(connector, aiohttp.TCPConnector)
|
2017-01-30 00:15:40 +00:00
|
|
|
|
|
|
|
|
2023-10-24 23:40:39 +00:00
|
|
|
@pytest.mark.parametrize(
|
|
|
|
("verify_ssl", "expected_family"),
|
|
|
|
[(True, 0), (False, 0), (True, 4), (False, 4), (True, 6), (False, 6)],
|
|
|
|
)
|
|
|
|
async def test_get_clientsession_cleanup(
|
|
|
|
hass: HomeAssistant, verify_ssl: bool, expected_family: int
|
|
|
|
) -> None:
|
|
|
|
"""Test init clientsession cleanup."""
|
|
|
|
client.async_get_clientsession(hass, verify_ssl=verify_ssl, family=expected_family)
|
2017-01-30 12:09:36 +00:00
|
|
|
|
2023-10-24 23:40:39 +00:00
|
|
|
client_session = hass.data[client.DATA_CLIENTSESSION][(verify_ssl, expected_family)]
|
|
|
|
assert isinstance(client_session, aiohttp.ClientSession)
|
|
|
|
connector = hass.data[client.DATA_CONNECTOR][(verify_ssl, expected_family)]
|
|
|
|
assert isinstance(connector, aiohttp.TCPConnector)
|
2017-01-30 12:09:36 +00:00
|
|
|
|
2020-04-07 16:33:23 +00:00
|
|
|
hass.bus.async_fire(EVENT_HOMEASSISTANT_CLOSE)
|
|
|
|
await hass.async_block_till_done()
|
2017-01-30 12:09:36 +00:00
|
|
|
|
2023-10-24 23:40:39 +00:00
|
|
|
assert client_session.closed
|
|
|
|
assert connector.closed
|
2017-01-30 12:09:36 +00:00
|
|
|
|
2017-01-30 00:15:40 +00:00
|
|
|
|
2023-02-08 07:51:43 +00:00
|
|
|
async def test_get_clientsession_patched_close(hass: HomeAssistant) -> None:
|
2020-05-13 07:58:33 +00:00
|
|
|
"""Test closing clientsession does not work."""
|
2023-10-24 23:40:39 +00:00
|
|
|
|
|
|
|
verify_ssl = True
|
|
|
|
family = 0
|
|
|
|
|
2020-05-13 07:58:33 +00:00
|
|
|
with patch("aiohttp.ClientSession.close") as mock_close:
|
|
|
|
session = client.async_get_clientsession(hass)
|
|
|
|
|
2023-10-24 23:40:39 +00:00
|
|
|
assert isinstance(
|
|
|
|
hass.data[client.DATA_CLIENTSESSION][(verify_ssl, family)],
|
|
|
|
aiohttp.ClientSession,
|
|
|
|
)
|
|
|
|
assert isinstance(
|
|
|
|
hass.data[client.DATA_CONNECTOR][(verify_ssl, family)], aiohttp.TCPConnector
|
|
|
|
)
|
2020-05-13 07:58:33 +00:00
|
|
|
|
|
|
|
with pytest.raises(RuntimeError):
|
|
|
|
await session.close()
|
|
|
|
|
|
|
|
assert mock_close.call_count == 0
|
|
|
|
|
|
|
|
|
2021-12-08 09:50:56 +00:00
|
|
|
@patch("homeassistant.helpers.frame._REPORTED_INTEGRATIONS", set())
|
2023-02-20 10:42:56 +00:00
|
|
|
async def test_warning_close_session_integration(
|
|
|
|
hass: HomeAssistant, caplog: pytest.LogCaptureFixture
|
|
|
|
) -> None:
|
2020-05-13 07:58:33 +00:00
|
|
|
"""Test log warning message when closing the session from integration context."""
|
|
|
|
with patch(
|
|
|
|
"homeassistant.helpers.frame.extract_stack",
|
|
|
|
return_value=[
|
|
|
|
Mock(
|
|
|
|
filename="/home/paulus/homeassistant/core.py",
|
|
|
|
lineno="23",
|
|
|
|
line="do_something()",
|
|
|
|
),
|
|
|
|
Mock(
|
|
|
|
filename="/home/paulus/homeassistant/components/hue/light.py",
|
|
|
|
lineno="23",
|
|
|
|
line="await session.close()",
|
|
|
|
),
|
|
|
|
Mock(
|
|
|
|
filename="/home/paulus/aiohue/lights.py",
|
|
|
|
lineno="2",
|
|
|
|
line="something()",
|
|
|
|
),
|
|
|
|
],
|
|
|
|
):
|
|
|
|
session = client.async_get_clientsession(hass)
|
|
|
|
await session.close()
|
|
|
|
assert (
|
2023-10-05 17:52:26 +00:00
|
|
|
"Detected that integration 'hue' closes the Home Assistant aiohttp session at "
|
|
|
|
"homeassistant/components/hue/light.py, line 23: await session.close(), "
|
|
|
|
"please create a bug report at https://github.com/home-assistant/core/issues?"
|
|
|
|
"q=is%3Aopen+is%3Aissue+label%3A%22integration%3A+hue%22"
|
2020-05-13 07:58:33 +00:00
|
|
|
) in caplog.text
|
|
|
|
|
|
|
|
|
2021-12-08 09:50:56 +00:00
|
|
|
@patch("homeassistant.helpers.frame._REPORTED_INTEGRATIONS", set())
|
2023-02-20 10:42:56 +00:00
|
|
|
async def test_warning_close_session_custom(
|
|
|
|
hass: HomeAssistant, caplog: pytest.LogCaptureFixture
|
|
|
|
) -> None:
|
2020-05-13 07:58:33 +00:00
|
|
|
"""Test log warning message when closing the session from custom context."""
|
2023-10-05 17:52:26 +00:00
|
|
|
mock_integration(hass, MockModule("hue"), built_in=False)
|
2020-05-13 07:58:33 +00:00
|
|
|
with patch(
|
|
|
|
"homeassistant.helpers.frame.extract_stack",
|
|
|
|
return_value=[
|
|
|
|
Mock(
|
|
|
|
filename="/home/paulus/homeassistant/core.py",
|
|
|
|
lineno="23",
|
|
|
|
line="do_something()",
|
|
|
|
),
|
|
|
|
Mock(
|
|
|
|
filename="/home/paulus/config/custom_components/hue/light.py",
|
|
|
|
lineno="23",
|
|
|
|
line="await session.close()",
|
|
|
|
),
|
|
|
|
Mock(
|
|
|
|
filename="/home/paulus/aiohue/lights.py",
|
|
|
|
lineno="2",
|
|
|
|
line="something()",
|
|
|
|
),
|
|
|
|
],
|
|
|
|
):
|
|
|
|
session = client.async_get_clientsession(hass)
|
|
|
|
await session.close()
|
|
|
|
assert (
|
2023-10-05 17:52:26 +00:00
|
|
|
"Detected that custom integration 'hue' closes the Home Assistant aiohttp "
|
|
|
|
"session at custom_components/hue/light.py, line 23: await session.close(), "
|
|
|
|
"please report it to the author of the 'hue' custom integration"
|
|
|
|
) in caplog.text
|
2020-05-13 07:58:33 +00:00
|
|
|
|
|
|
|
|
2023-02-20 10:42:56 +00:00
|
|
|
async def test_async_aiohttp_proxy_stream(
|
|
|
|
aioclient_mock: AiohttpClientMocker, camera_client
|
|
|
|
) -> None:
|
2017-01-30 00:15:40 +00:00
|
|
|
"""Test that it fetches the given url."""
|
2019-07-31 19:25:30 +00:00
|
|
|
aioclient_mock.get("http://example.com/mjpeg_stream", content=b"Frame1Frame2Frame3")
|
2017-01-30 00:15:40 +00:00
|
|
|
|
2022-04-04 15:57:48 +00:00
|
|
|
resp = await camera_client.get("/api/camera_proxy_stream/camera.mjpeg_camera")
|
2017-01-30 00:15:40 +00:00
|
|
|
|
|
|
|
assert resp.status == 200
|
|
|
|
assert aioclient_mock.call_count == 1
|
2020-04-07 16:33:23 +00:00
|
|
|
body = await resp.text()
|
2019-07-31 19:25:30 +00:00
|
|
|
assert body == "Frame1Frame2Frame3"
|
2017-01-30 00:15:40 +00:00
|
|
|
|
2018-03-05 21:28:41 +00:00
|
|
|
|
2023-02-20 10:42:56 +00:00
|
|
|
async def test_async_aiohttp_proxy_stream_timeout(
|
|
|
|
aioclient_mock: AiohttpClientMocker, camera_client
|
|
|
|
) -> None:
|
2018-03-05 21:28:41 +00:00
|
|
|
"""Test that it fetches the given url."""
|
2019-07-31 19:25:30 +00:00
|
|
|
aioclient_mock.get("http://example.com/mjpeg_stream", exc=asyncio.TimeoutError())
|
2017-01-30 00:15:40 +00:00
|
|
|
|
2022-04-04 15:57:48 +00:00
|
|
|
resp = await camera_client.get("/api/camera_proxy_stream/camera.mjpeg_camera")
|
2017-03-30 07:50:53 +00:00
|
|
|
assert resp.status == 504
|
2017-01-30 00:15:40 +00:00
|
|
|
|
2018-03-05 21:28:41 +00:00
|
|
|
|
2023-02-20 10:42:56 +00:00
|
|
|
async def test_async_aiohttp_proxy_stream_client_err(
|
|
|
|
aioclient_mock: AiohttpClientMocker, camera_client
|
|
|
|
) -> None:
|
2018-03-05 21:28:41 +00:00
|
|
|
"""Test that it fetches the given url."""
|
2019-07-31 19:25:30 +00:00
|
|
|
aioclient_mock.get("http://example.com/mjpeg_stream", exc=aiohttp.ClientError())
|
2017-01-30 00:15:40 +00:00
|
|
|
|
2022-04-04 15:57:48 +00:00
|
|
|
resp = await camera_client.get("/api/camera_proxy_stream/camera.mjpeg_camera")
|
2017-03-30 07:50:53 +00:00
|
|
|
assert resp.status == 502
|
2021-06-04 16:14:18 +00:00
|
|
|
|
|
|
|
|
2023-02-20 10:42:56 +00:00
|
|
|
async def test_sending_named_tuple(
|
|
|
|
hass: HomeAssistant, aioclient_mock: AiohttpClientMocker
|
|
|
|
) -> None:
|
2022-07-11 21:46:55 +00:00
|
|
|
"""Test sending a named tuple in json."""
|
|
|
|
resp = aioclient_mock.post("http://127.0.0.1/rgb", json={"rgb": RGBColor(4, 3, 2)})
|
|
|
|
session = client.async_create_clientsession(hass)
|
|
|
|
resp = await session.post("http://127.0.0.1/rgb", json={"rgb": RGBColor(4, 3, 2)})
|
|
|
|
assert resp.status == 200
|
|
|
|
await resp.json() == {"rgb": RGBColor(4, 3, 2)}
|
|
|
|
aioclient_mock.mock_calls[0][2]["rgb"] == RGBColor(4, 3, 2)
|
|
|
|
|
|
|
|
|
2023-02-08 07:51:43 +00:00
|
|
|
async def test_client_session_immutable_headers(hass: HomeAssistant) -> None:
|
2021-06-04 16:14:18 +00:00
|
|
|
"""Test we can't mutate headers."""
|
|
|
|
session = client.async_get_clientsession(hass)
|
|
|
|
|
|
|
|
with pytest.raises(TypeError):
|
|
|
|
session.headers["user-agent"] = "bla"
|
|
|
|
|
|
|
|
with pytest.raises(AttributeError):
|
|
|
|
session.headers.update({"user-agent": "bla"})
|