"""Test stream init.""" import logging from typing import Any from unittest.mock import MagicMock, patch import av import pytest from homeassistant.components.stream import ( SOURCE_TIMEOUT, StreamClientError, StreamOpenClientError, __name__ as stream_name, async_check_stream_client_error, create_stream, ) from homeassistant.components.stream.const import ATTR_PREFER_TCP from homeassistant.const import EVENT_LOGGING_CHANGED from homeassistant.core import HomeAssistant from homeassistant.exceptions import HomeAssistantError from homeassistant.setup import async_setup_component from .common import dynamic_stream_settings async def test_stream_not_setup(hass: HomeAssistant, h264_video) -> None: """Test hls stream. Purposefully not mocking anything here to test full integration with the stream component. """ with pytest.raises(HomeAssistantError, match="Stream integration is not set up"): create_stream(hass, "rtsp://foobar", {}, dynamic_stream_settings()) with pytest.raises(HomeAssistantError, match="Stream integration is not set up"): await async_check_stream_client_error(hass, "rtsp://foobar") async def test_log_levels( hass: HomeAssistant, caplog: pytest.LogCaptureFixture ) -> None: """Test that the worker logs the url without username and password.""" await async_setup_component(hass, "stream", {"stream": {}}) # These namespaces should only pass log messages when the stream logger # is at logging.DEBUG or below namespaces_to_toggle = ( "mp4", "h264", "hevc", "rtsp", "tcp", "tls", "mpegts", "NULL", ) logging.getLogger(stream_name).setLevel(logging.INFO) hass.bus.async_fire(EVENT_LOGGING_CHANGED) await hass.async_block_till_done() # Since logging is at INFO, these should not pass for namespace in namespaces_to_toggle: av.logging.log(av.logging.ERROR, namespace, "SHOULD NOT PASS") logging.getLogger(stream_name).setLevel(logging.DEBUG) hass.bus.async_fire(EVENT_LOGGING_CHANGED) await hass.async_block_till_done() # Since logging is now at DEBUG, these should now pass for namespace in namespaces_to_toggle: av.logging.log(av.logging.ERROR, namespace, "SHOULD PASS") # Even though logging is at DEBUG, these should not pass av.logging.log(av.logging.WARNING, "mp4", "SHOULD NOT PASS") av.logging.log(av.logging.WARNING, "swscaler", "SHOULD NOT PASS") assert "SHOULD PASS" in caplog.text assert "SHOULD NOT PASS" not in caplog.text async def test_check_open_stream_params(hass: HomeAssistant) -> None: """Test check open stream params.""" await async_setup_component(hass, "stream", {"stream": {}}) container_mock = MagicMock() source = "rtsp://foobar" with patch("av.open", return_value=container_mock) as open_mock: await async_check_stream_client_error(hass, source) options = { "rtsp_flags": ATTR_PREFER_TCP, "stimeout": "5000000", } open_mock.assert_called_once_with(source, options=options, timeout=SOURCE_TIMEOUT) container_mock.close.assert_called_once() container_mock.reset_mock() with ( patch("av.open", return_value=container_mock) as open_mock, pytest.raises(HomeAssistantError, match="Invalid stream options"), ): await async_check_stream_client_error(hass, source, {"foo": "bar"}) @pytest.mark.parametrize( ("error", "enum_result"), [ pytest.param( av.HTTPBadRequestError(400, ""), StreamClientError.BadRequest, id="BadRequest", ), pytest.param( av.HTTPUnauthorizedError(401, ""), StreamClientError.Unauthorized, id="Unauthorized", ), pytest.param( av.HTTPForbiddenError(403, ""), StreamClientError.Forbidden, id="Forbidden" ), pytest.param( av.HTTPNotFoundError(404, ""), StreamClientError.NotFound, id="NotFound" ), pytest.param( av.HTTPOtherClientError(408, ""), StreamClientError.Other, id="Other" ), ], ) async def test_try_open_stream_error( hass: HomeAssistant, error: av.HTTPClientError, enum_result: StreamClientError ) -> None: """Test trying to open a stream.""" await async_setup_component(hass, "stream", {"stream": {}}) with ( patch("av.open", side_effect=error), pytest.raises(StreamOpenClientError) as ex, ): await async_check_stream_client_error(hass, "rtsp://foobar") assert ex.value.error_code is enum_result @pytest.mark.parametrize( ("options", "expected_pyav_options"), [ ( {}, {"rtsp_flags": "prefer_tcp", "stimeout": "5000000"}, ), ( {"rtsp_transport": "udp"}, { "rtsp_flags": "prefer_tcp", "rtsp_transport": "udp", "stimeout": "5000000", }, ), ( {"use_wallclock_as_timestamps": True}, { "rtsp_flags": "prefer_tcp", "stimeout": "5000000", "use_wallclock_as_timestamps": "1", }, ), ], ) async def test_convert_stream_options( hass: HomeAssistant, options: dict[str, Any], expected_pyav_options: dict[str, Any], ) -> None: """Test stream options.""" await async_setup_component(hass, "stream", {"stream": {}}) container_mock = MagicMock() source = "rtsp://foobar" with patch("av.open", return_value=container_mock) as open_mock: await async_check_stream_client_error(hass, source, options) open_mock.assert_called_once_with( source, options=expected_pyav_options, timeout=SOURCE_TIMEOUT ) container_mock.close.assert_called_once()