core/tests/components/media_player/test_init.py

557 lines
18 KiB
Python
Raw Normal View History

"""Test the base functions of the media player."""
from enum import Enum
from http import HTTPStatus
from types import ModuleType
2021-01-01 21:31:56 +00:00
from unittest.mock import patch
import pytest
import voluptuous as vol
from homeassistant.components import media_player
from homeassistant.components.media_player import (
ATTR_MEDIA_CONTENT_ID,
ATTR_MEDIA_CONTENT_TYPE,
BrowseMedia,
MediaClass,
MediaPlayerEnqueue,
MediaPlayerEntity,
MediaPlayerEntityFeature,
)
from homeassistant.components.media_player.const import SERVICE_BROWSE_MEDIA
from homeassistant.components.websocket_api import TYPE_RESULT
from homeassistant.const import ATTR_ENTITY_ID, STATE_OFF
from homeassistant.core import HomeAssistant
from homeassistant.setup import async_setup_component
from tests.common import (
MockEntityPlatform,
help_test_all,
import_and_test_deprecated_constant_enum,
)
from tests.test_util.aiohttp import AiohttpClientMocker
from tests.typing import ClientSessionGenerator, WebSocketGenerator
@pytest.fixture(autouse=True)
async def setup_homeassistant(hass: HomeAssistant):
"""Set up the homeassistant integration."""
await async_setup_component(hass, "homeassistant", {})
def _create_tuples(enum: type[Enum], constant_prefix: str) -> list[tuple[Enum, str]]:
return [
(enum_field, constant_prefix)
for enum_field in enum
if enum_field
not in [
MediaPlayerEntityFeature.MEDIA_ANNOUNCE,
MediaPlayerEntityFeature.MEDIA_ENQUEUE,
]
]
@pytest.mark.parametrize(
"module",
[media_player, media_player.const],
)
def test_all(module: ModuleType) -> None:
"""Test module.__all__ is correctly set."""
help_test_all(module)
@pytest.mark.parametrize(
("enum", "constant_prefix"),
_create_tuples(media_player.MediaPlayerEntityFeature, "SUPPORT_")
+ _create_tuples(media_player.MediaPlayerDeviceClass, "DEVICE_CLASS_"),
)
@pytest.mark.parametrize(
"module",
[media_player],
)
def test_deprecated_constants(
caplog: pytest.LogCaptureFixture,
enum: Enum,
constant_prefix: str,
module: ModuleType,
) -> None:
"""Test deprecated constants."""
import_and_test_deprecated_constant_enum(
caplog, module, enum, constant_prefix, "2025.10"
)
@pytest.mark.parametrize(
("enum", "constant_prefix"),
_create_tuples(media_player.MediaClass, "MEDIA_CLASS_")
+ _create_tuples(media_player.MediaPlayerEntityFeature, "SUPPORT_")
+ _create_tuples(media_player.MediaType, "MEDIA_TYPE_")
+ _create_tuples(media_player.RepeatMode, "REPEAT_MODE_"),
)
@pytest.mark.parametrize(
"module",
[media_player.const],
)
def test_deprecated_constants_const(
caplog: pytest.LogCaptureFixture,
enum: Enum,
constant_prefix: str,
module: ModuleType,
) -> None:
"""Test deprecated constants."""
import_and_test_deprecated_constant_enum(
caplog, module, enum, constant_prefix, "2025.10"
)
@pytest.mark.parametrize(
"property_suffix",
[
"play",
"pause",
"stop",
"seek",
"volume_set",
"volume_mute",
"previous_track",
"next_track",
"play_media",
"select_source",
"select_sound_mode",
"clear_playlist",
"shuffle_set",
"grouping",
],
)
def test_support_properties(hass: HomeAssistant, property_suffix: str) -> None:
"""Test support_*** properties explicitly."""
all_features = media_player.MediaPlayerEntityFeature(653887)
feature = media_player.MediaPlayerEntityFeature[property_suffix.upper()]
entity1 = MediaPlayerEntity()
entity1.hass = hass
entity1.platform = MockEntityPlatform(hass)
entity1._attr_supported_features = media_player.MediaPlayerEntityFeature(0)
entity2 = MediaPlayerEntity()
entity2.hass = hass
entity2.platform = MockEntityPlatform(hass)
entity2._attr_supported_features = all_features
entity3 = MediaPlayerEntity()
entity3.hass = hass
entity3.platform = MockEntityPlatform(hass)
entity3._attr_supported_features = feature
entity4 = MediaPlayerEntity()
entity4.hass = hass
entity4.platform = MockEntityPlatform(hass)
entity4._attr_supported_features = all_features - feature
assert getattr(entity1, f"support_{property_suffix}") is False
assert getattr(entity2, f"support_{property_suffix}") is True
assert getattr(entity3, f"support_{property_suffix}") is True
assert getattr(entity4, f"support_{property_suffix}") is False
async def test_get_image_http(
hass: HomeAssistant, hass_client_no_auth: ClientSessionGenerator
) -> None:
"""Test get image via http command."""
2019-07-31 19:25:30 +00:00
await async_setup_component(
hass, "media_player", {"media_player": {"platform": "demo"}}
)
await hass.async_block_till_done()
state = hass.states.get("media_player.bedroom")
assert "entity_picture_local" not in state.attributes
client = await hass_client_no_auth()
2019-07-31 19:25:30 +00:00
with patch(
"homeassistant.components.media_player.MediaPlayerEntity.async_get_media_image",
return_value=(b"image", "image/jpeg"),
2019-07-31 19:25:30 +00:00
):
resp = await client.get(state.attributes["entity_picture"])
content = await resp.read()
2019-07-31 19:25:30 +00:00
assert content == b"image"
async def test_get_image_http_remote(
hass: HomeAssistant, hass_client_no_auth: ClientSessionGenerator
) -> None:
"""Test get image url via http command."""
2019-07-31 19:25:30 +00:00
with patch(
"homeassistant.components.media_player.MediaPlayerEntity."
2019-07-31 19:25:30 +00:00
"media_image_remotely_accessible",
return_value=True,
):
await async_setup_component(
hass, "media_player", {"media_player": {"platform": "demo"}}
2019-07-31 19:25:30 +00:00
)
await hass.async_block_till_done()
state = hass.states.get("media_player.bedroom")
assert "entity_picture_local" in state.attributes
client = await hass_client_no_auth()
with patch(
"homeassistant.components.media_player.MediaPlayerEntity."
"async_get_media_image",
return_value=(b"image", "image/jpeg"),
):
resp = await client.get(state.attributes["entity_picture_local"])
content = await resp.read()
assert content == b"image"
async def test_get_image_http_log_credentials_redacted(
hass: HomeAssistant,
hass_client_no_auth: ClientSessionGenerator,
aioclient_mock: AiohttpClientMocker,
caplog: pytest.LogCaptureFixture,
) -> None:
"""Test credentials are redacted when logging url when fetching image."""
url = "http://vi:pass@example.com/default.jpg"
with patch(
"homeassistant.components.demo.media_player.DemoYoutubePlayer.media_image_url",
url,
):
await async_setup_component(
hass, "media_player", {"media_player": {"platform": "demo"}}
)
await hass.async_block_till_done()
state = hass.states.get("media_player.bedroom")
assert "entity_picture_local" not in state.attributes
aioclient_mock.get(url, exc=TimeoutError())
client = await hass_client_no_auth()
resp = await client.get(state.attributes["entity_picture"])
assert resp.status == HTTPStatus.INTERNAL_SERVER_ERROR
assert f"Error retrieving proxied image from {url}" not in caplog.text
assert (
"Error retrieving proxied image from "
f"{url.replace('pass', 'xxxxxxxx').replace('vi', 'xxxx')}"
) in caplog.text
async def test_get_async_get_browse_image(
hass: HomeAssistant,
hass_client_no_auth: ClientSessionGenerator,
hass_ws_client: WebSocketGenerator,
) -> None:
"""Test get browse image."""
await async_setup_component(
hass, "media_player", {"media_player": {"platform": "demo"}}
)
await hass.async_block_till_done()
entity_comp = hass.data.get("entity_components", {}).get("media_player")
assert entity_comp
player = entity_comp.get_entity("media_player.bedroom")
assert player
client = await hass_client_no_auth()
with patch(
"homeassistant.components.media_player.MediaPlayerEntity."
"async_get_browse_image",
return_value=(b"image", "image/jpeg"),
):
url = player.get_browse_image_url("album", "abcd")
resp = await client.get(url)
content = await resp.read()
assert content == b"image"
async def test_media_browse(
hass: HomeAssistant, hass_ws_client: WebSocketGenerator
) -> None:
"""Test browsing media."""
await async_setup_component(
hass, "media_player", {"media_player": {"platform": "demo"}}
)
await hass.async_block_till_done()
client = await hass_ws_client(hass)
with patch(
"homeassistant.components.demo.media_player.DemoBrowsePlayer.async_browse_media",
return_value=BrowseMedia(
media_class=MediaClass.DIRECTORY,
media_content_id="mock-id",
media_content_type="mock-type",
title="Mock Title",
can_play=False,
can_expand=True,
),
) as mock_browse_media:
await client.send_json(
{
"id": 5,
"type": "media_player/browse_media",
"entity_id": "media_player.browse",
"media_content_type": "album",
"media_content_id": "abcd",
}
)
msg = await client.receive_json()
assert msg["id"] == 5
assert msg["type"] == TYPE_RESULT
assert msg["success"]
assert msg["result"] == {
"title": "Mock Title",
"media_class": "directory",
"media_content_type": "mock-type",
"media_content_id": "mock-id",
"can_play": False,
"can_expand": True,
"children_media_class": None,
"thumbnail": None,
"not_shown": 0,
"children": [],
}
assert mock_browse_media.mock_calls[0][1] == ("album", "abcd")
with patch(
"homeassistant.components.demo.media_player.DemoBrowsePlayer.async_browse_media",
return_value={"bla": "yo"},
):
await client.send_json(
{
"id": 6,
"type": "media_player/browse_media",
"entity_id": "media_player.browse",
}
)
msg = await client.receive_json()
assert msg["id"] == 6
assert msg["type"] == TYPE_RESULT
assert msg["success"]
assert msg["result"] == {"bla": "yo"}
async def test_media_browse_service(hass: HomeAssistant) -> None:
"""Test browsing media using service call."""
await async_setup_component(
hass, "media_player", {"media_player": {"platform": "demo"}}
)
await hass.async_block_till_done()
with patch(
"homeassistant.components.demo.media_player.DemoBrowsePlayer.async_browse_media",
return_value=BrowseMedia(
media_class=MediaClass.DIRECTORY,
media_content_id="mock-id",
media_content_type="mock-type",
title="Mock Title",
can_play=False,
can_expand=True,
children=[
BrowseMedia(
media_class=MediaClass.ALBUM,
media_content_id="album1 content id",
media_content_type="album",
title="Album 1",
can_play=True,
can_expand=True,
),
BrowseMedia(
media_class=MediaClass.ALBUM,
media_content_id="album2 content id",
media_content_type="album",
title="Album 2",
can_play=True,
can_expand=True,
),
],
),
) as mock_browse_media:
result = await hass.services.async_call(
"media_player",
SERVICE_BROWSE_MEDIA,
{
ATTR_ENTITY_ID: "media_player.browse",
ATTR_MEDIA_CONTENT_TYPE: "album",
ATTR_MEDIA_CONTENT_ID: "title=Album*",
},
blocking=True,
return_response=True,
)
mock_browse_media.assert_called_with(
media_content_type="album", media_content_id="title=Album*"
)
browse_res: BrowseMedia = result["media_player.browse"]
assert browse_res.title == "Mock Title"
assert browse_res.media_class == "directory"
assert browse_res.media_content_type == "mock-type"
assert browse_res.media_content_id == "mock-id"
assert browse_res.can_play is False
assert browse_res.can_expand is True
assert len(browse_res.children) == 2
assert browse_res.children[0].title == "Album 1"
assert browse_res.children[0].media_class == "album"
assert browse_res.children[0].media_content_id == "album1 content id"
assert browse_res.children[0].media_content_type == "album"
assert browse_res.children[1].title == "Album 2"
assert browse_res.children[1].media_class == "album"
assert browse_res.children[1].media_content_id == "album2 content id"
assert browse_res.children[1].media_content_type == "album"
async def test_group_members_available_when_off(hass: HomeAssistant) -> None:
"""Test that group_members are still available when media_player is off."""
await async_setup_component(
hass, "media_player", {"media_player": {"platform": "demo"}}
)
await hass.async_block_till_done()
await hass.services.async_call(
"media_player",
"turn_off",
{ATTR_ENTITY_ID: "media_player.group"},
blocking=True,
)
state = hass.states.get("media_player.group")
assert state.state == STATE_OFF
assert "group_members" in state.attributes
@pytest.mark.parametrize(
("input", "expected"),
[
(True, MediaPlayerEnqueue.ADD),
(False, MediaPlayerEnqueue.PLAY),
("play", MediaPlayerEnqueue.PLAY),
("next", MediaPlayerEnqueue.NEXT),
("add", MediaPlayerEnqueue.ADD),
("replace", MediaPlayerEnqueue.REPLACE),
],
)
async def test_enqueue_rewrite(hass: HomeAssistant, input, expected) -> None:
"""Test that group_members are still available when media_player is off."""
await async_setup_component(
hass, "media_player", {"media_player": {"platform": "demo"}}
)
await hass.async_block_till_done()
# Fake group support for DemoYoutubePlayer
with patch(
"homeassistant.components.demo.media_player.DemoYoutubePlayer.play_media",
) as mock_play_media:
await hass.services.async_call(
"media_player",
"play_media",
{
"entity_id": "media_player.bedroom",
"media_content_type": "music",
"media_content_id": "1234",
"enqueue": input,
},
blocking=True,
)
assert len(mock_play_media.mock_calls) == 1
assert mock_play_media.mock_calls[0][2]["enqueue"] == expected
async def test_enqueue_alert_exclusive(hass: HomeAssistant) -> None:
"""Test that alert and enqueue cannot be used together."""
await async_setup_component(
hass, "media_player", {"media_player": {"platform": "demo"}}
)
await hass.async_block_till_done()
with pytest.raises(vol.Invalid):
await hass.services.async_call(
"media_player",
"play_media",
{
"entity_id": "media_player.bedroom",
"media_content_type": "music",
"media_content_id": "1234",
"enqueue": "play",
"announce": True,
},
blocking=True,
)
2024-09-24 06:51:08 +00:00
@pytest.mark.parametrize(
"media_content_id",
[
"a/b c/d+e%2Fg{}",
"a/b c/d+e%2D",
"a/b c/d+e%2E",
"2012-06%20Pool%20party%20%2F%20BBQ",
],
)
async def test_get_async_get_browse_image_quoting(
hass: HomeAssistant,
hass_client_no_auth: ClientSessionGenerator,
hass_ws_client: WebSocketGenerator,
2024-09-24 06:51:08 +00:00
media_content_id: str,
) -> None:
"""Test get browse image using media_content_id with special characters.
async_get_browse_image() should get called with the same string that is
passed into get_browse_image_url().
"""
await async_setup_component(
hass, "media_player", {"media_player": {"platform": "demo"}}
)
await hass.async_block_till_done()
entity_comp = hass.data.get("entity_components", {}).get("media_player")
assert entity_comp
player = entity_comp.get_entity("media_player.bedroom")
assert player
client = await hass_client_no_auth()
with patch(
"homeassistant.components.media_player.MediaPlayerEntity."
"async_get_browse_image",
) as mock_browse_image:
url = player.get_browse_image_url("album", media_content_id)
await client.get(url)
mock_browse_image.assert_called_with("album", media_content_id, None)
def test_deprecated_supported_features_ints(
hass: HomeAssistant, caplog: pytest.LogCaptureFixture
) -> None:
"""Test deprecated supported features ints."""
class MockMediaPlayerEntity(MediaPlayerEntity):
@property
def supported_features(self) -> int:
"""Return supported features."""
return 1
entity = MockMediaPlayerEntity()
entity.hass = hass
entity.platform = MockEntityPlatform(hass)
assert entity.supported_features_compat is MediaPlayerEntityFeature(1)
assert "MockMediaPlayerEntity" in caplog.text
assert "is using deprecated supported features values" in caplog.text
assert "Instead it should use" in caplog.text
assert "MediaPlayerEntityFeature.PAUSE" in caplog.text
caplog.clear()
assert entity.supported_features_compat is MediaPlayerEntityFeature(1)
assert "is using deprecated supported features values" not in caplog.text