376 lines
14 KiB
Python
376 lines
14 KiB
Python
"""The tests for the DirecTV Media player platform."""
|
|
from __future__ import annotations
|
|
|
|
from datetime import datetime, timedelta
|
|
from unittest.mock import patch
|
|
|
|
import pytest
|
|
|
|
from homeassistant.components.directv.media_player import (
|
|
ATTR_MEDIA_CURRENTLY_RECORDING,
|
|
ATTR_MEDIA_RATING,
|
|
ATTR_MEDIA_RECORDED,
|
|
ATTR_MEDIA_START_TIME,
|
|
)
|
|
from homeassistant.components.media_player import (
|
|
ATTR_INPUT_SOURCE,
|
|
ATTR_MEDIA_ALBUM_NAME,
|
|
ATTR_MEDIA_ARTIST,
|
|
ATTR_MEDIA_CHANNEL,
|
|
ATTR_MEDIA_CONTENT_ID,
|
|
ATTR_MEDIA_CONTENT_TYPE,
|
|
ATTR_MEDIA_DURATION,
|
|
ATTR_MEDIA_ENQUEUE,
|
|
ATTR_MEDIA_POSITION,
|
|
ATTR_MEDIA_POSITION_UPDATED_AT,
|
|
ATTR_MEDIA_SERIES_TITLE,
|
|
ATTR_MEDIA_TITLE,
|
|
DOMAIN as MP_DOMAIN,
|
|
SERVICE_PLAY_MEDIA,
|
|
MediaPlayerDeviceClass,
|
|
MediaPlayerEntityFeature,
|
|
MediaType,
|
|
)
|
|
from homeassistant.const import (
|
|
ATTR_ENTITY_ID,
|
|
SERVICE_MEDIA_NEXT_TRACK,
|
|
SERVICE_MEDIA_PAUSE,
|
|
SERVICE_MEDIA_PLAY,
|
|
SERVICE_MEDIA_PREVIOUS_TRACK,
|
|
SERVICE_MEDIA_STOP,
|
|
SERVICE_TURN_OFF,
|
|
SERVICE_TURN_ON,
|
|
STATE_OFF,
|
|
STATE_PAUSED,
|
|
STATE_PLAYING,
|
|
STATE_UNAVAILABLE,
|
|
)
|
|
from homeassistant.core import HomeAssistant
|
|
from homeassistant.helpers import entity_registry as er
|
|
from homeassistant.util import dt as dt_util
|
|
|
|
from . import setup_integration
|
|
|
|
from tests.test_util.aiohttp import AiohttpClientMocker
|
|
|
|
ATTR_UNIQUE_ID = "unique_id"
|
|
CLIENT_ENTITY_ID = f"{MP_DOMAIN}.client"
|
|
MAIN_ENTITY_ID = f"{MP_DOMAIN}.host"
|
|
MUSIC_ENTITY_ID = f"{MP_DOMAIN}.music_client"
|
|
RESTRICTED_ENTITY_ID = f"{MP_DOMAIN}.restricted_client"
|
|
STANDBY_ENTITY_ID = f"{MP_DOMAIN}.standby_client"
|
|
UNAVAILABLE_ENTITY_ID = f"{MP_DOMAIN}.unavailable_client"
|
|
|
|
|
|
@pytest.fixture
|
|
def mock_now() -> datetime:
|
|
"""Fixture for dtutil.now."""
|
|
return dt_util.utcnow()
|
|
|
|
|
|
async def async_turn_on(hass: HomeAssistant, entity_id: str | None = None) -> None:
|
|
"""Turn on specified media player or all."""
|
|
data = {ATTR_ENTITY_ID: entity_id} if entity_id else {}
|
|
await hass.services.async_call(MP_DOMAIN, SERVICE_TURN_ON, data)
|
|
|
|
|
|
async def async_turn_off(hass: HomeAssistant, entity_id: str | None = None) -> None:
|
|
"""Turn off specified media player or all."""
|
|
data = {ATTR_ENTITY_ID: entity_id} if entity_id else {}
|
|
await hass.services.async_call(MP_DOMAIN, SERVICE_TURN_OFF, data)
|
|
|
|
|
|
async def async_media_pause(hass: HomeAssistant, entity_id: str | None = None) -> None:
|
|
"""Send the media player the command for pause."""
|
|
data = {ATTR_ENTITY_ID: entity_id} if entity_id else {}
|
|
await hass.services.async_call(MP_DOMAIN, SERVICE_MEDIA_PAUSE, data)
|
|
|
|
|
|
async def async_media_play(hass: HomeAssistant, entity_id: str | None = None) -> None:
|
|
"""Send the media player the command for play/pause."""
|
|
data = {ATTR_ENTITY_ID: entity_id} if entity_id else {}
|
|
await hass.services.async_call(MP_DOMAIN, SERVICE_MEDIA_PLAY, data)
|
|
|
|
|
|
async def async_media_stop(hass: HomeAssistant, entity_id: str | None = None) -> None:
|
|
"""Send the media player the command for stop."""
|
|
data = {ATTR_ENTITY_ID: entity_id} if entity_id else {}
|
|
await hass.services.async_call(MP_DOMAIN, SERVICE_MEDIA_STOP, data)
|
|
|
|
|
|
async def async_media_next_track(
|
|
hass: HomeAssistant, entity_id: str | None = None
|
|
) -> None:
|
|
"""Send the media player the command for next track."""
|
|
data = {ATTR_ENTITY_ID: entity_id} if entity_id else {}
|
|
await hass.services.async_call(MP_DOMAIN, SERVICE_MEDIA_NEXT_TRACK, data)
|
|
|
|
|
|
async def async_media_previous_track(
|
|
hass: HomeAssistant, entity_id: str | None = None
|
|
) -> None:
|
|
"""Send the media player the command for prev track."""
|
|
data = {ATTR_ENTITY_ID: entity_id} if entity_id else {}
|
|
await hass.services.async_call(MP_DOMAIN, SERVICE_MEDIA_PREVIOUS_TRACK, data)
|
|
|
|
|
|
async def async_play_media(
|
|
hass: HomeAssistant,
|
|
media_type: str,
|
|
media_id: str,
|
|
entity_id: str | None = None,
|
|
enqueue: str | None = None,
|
|
) -> None:
|
|
"""Send the media player the command for playing media."""
|
|
data = {ATTR_MEDIA_CONTENT_TYPE: media_type, ATTR_MEDIA_CONTENT_ID: media_id}
|
|
|
|
if entity_id:
|
|
data[ATTR_ENTITY_ID] = entity_id
|
|
|
|
if enqueue:
|
|
data[ATTR_MEDIA_ENQUEUE] = enqueue
|
|
|
|
await hass.services.async_call(MP_DOMAIN, SERVICE_PLAY_MEDIA, data)
|
|
|
|
|
|
async def test_setup(hass: HomeAssistant, aioclient_mock: AiohttpClientMocker) -> None:
|
|
"""Test setup with basic config."""
|
|
await setup_integration(hass, aioclient_mock)
|
|
assert hass.states.get(MAIN_ENTITY_ID)
|
|
assert hass.states.get(CLIENT_ENTITY_ID)
|
|
assert hass.states.get(UNAVAILABLE_ENTITY_ID)
|
|
|
|
|
|
async def test_unique_id(
|
|
hass: HomeAssistant, aioclient_mock: AiohttpClientMocker
|
|
) -> None:
|
|
"""Test unique id."""
|
|
await setup_integration(hass, aioclient_mock)
|
|
|
|
entity_registry = er.async_get(hass)
|
|
|
|
main = entity_registry.async_get(MAIN_ENTITY_ID)
|
|
assert main.original_device_class == MediaPlayerDeviceClass.RECEIVER
|
|
assert main.unique_id == "028877455858"
|
|
|
|
client = entity_registry.async_get(CLIENT_ENTITY_ID)
|
|
assert client.original_device_class == MediaPlayerDeviceClass.RECEIVER
|
|
assert client.unique_id == "2CA17D1CD30X"
|
|
|
|
unavailable_client = entity_registry.async_get(UNAVAILABLE_ENTITY_ID)
|
|
assert unavailable_client.original_device_class == MediaPlayerDeviceClass.RECEIVER
|
|
assert unavailable_client.unique_id == "9XXXXXXXXXX9"
|
|
|
|
|
|
async def test_supported_features(
|
|
hass: HomeAssistant, aioclient_mock: AiohttpClientMocker
|
|
) -> None:
|
|
"""Test supported features."""
|
|
await setup_integration(hass, aioclient_mock)
|
|
|
|
# Features supported for main DVR
|
|
state = hass.states.get(MAIN_ENTITY_ID)
|
|
assert (
|
|
MediaPlayerEntityFeature.PAUSE
|
|
| MediaPlayerEntityFeature.TURN_ON
|
|
| MediaPlayerEntityFeature.TURN_OFF
|
|
| MediaPlayerEntityFeature.PLAY_MEDIA
|
|
| MediaPlayerEntityFeature.STOP
|
|
| MediaPlayerEntityFeature.NEXT_TRACK
|
|
| MediaPlayerEntityFeature.PREVIOUS_TRACK
|
|
| MediaPlayerEntityFeature.PLAY
|
|
== state.attributes.get("supported_features")
|
|
)
|
|
|
|
# Feature supported for clients.
|
|
state = hass.states.get(CLIENT_ENTITY_ID)
|
|
assert (
|
|
MediaPlayerEntityFeature.PAUSE
|
|
| MediaPlayerEntityFeature.PLAY_MEDIA
|
|
| MediaPlayerEntityFeature.STOP
|
|
| MediaPlayerEntityFeature.NEXT_TRACK
|
|
| MediaPlayerEntityFeature.PREVIOUS_TRACK
|
|
| MediaPlayerEntityFeature.PLAY
|
|
== state.attributes.get("supported_features")
|
|
)
|
|
|
|
|
|
async def test_check_attributes(
|
|
hass: HomeAssistant,
|
|
mock_now: dt_util.dt.datetime,
|
|
aioclient_mock: AiohttpClientMocker,
|
|
) -> None:
|
|
"""Test attributes."""
|
|
await setup_integration(hass, aioclient_mock)
|
|
|
|
state = hass.states.get(MAIN_ENTITY_ID)
|
|
assert state.state == STATE_PLAYING
|
|
|
|
assert state.attributes.get(ATTR_MEDIA_CONTENT_ID) == "17016356"
|
|
assert state.attributes.get(ATTR_MEDIA_CONTENT_TYPE) == MediaType.MOVIE
|
|
assert state.attributes.get(ATTR_MEDIA_DURATION) == 7200
|
|
assert state.attributes.get(ATTR_MEDIA_POSITION) == 4437
|
|
assert state.attributes.get(ATTR_MEDIA_POSITION_UPDATED_AT)
|
|
assert state.attributes.get(ATTR_MEDIA_TITLE) == "Snow Bride"
|
|
assert state.attributes.get(ATTR_MEDIA_SERIES_TITLE) is None
|
|
assert state.attributes.get(ATTR_MEDIA_CHANNEL) == "{} ({})".format("HALLHD", "312")
|
|
assert state.attributes.get(ATTR_INPUT_SOURCE) == "312"
|
|
assert not state.attributes.get(ATTR_MEDIA_CURRENTLY_RECORDING)
|
|
assert state.attributes.get(ATTR_MEDIA_RATING) == "TV-G"
|
|
assert not state.attributes.get(ATTR_MEDIA_RECORDED)
|
|
assert state.attributes.get(ATTR_MEDIA_START_TIME) == datetime(
|
|
2020, 3, 21, 13, 0, tzinfo=dt_util.UTC
|
|
)
|
|
|
|
state = hass.states.get(CLIENT_ENTITY_ID)
|
|
assert state.state == STATE_PLAYING
|
|
|
|
assert state.attributes.get(ATTR_MEDIA_CONTENT_ID) == "4405732"
|
|
assert state.attributes.get(ATTR_MEDIA_CONTENT_TYPE) == MediaType.TVSHOW
|
|
assert state.attributes.get(ATTR_MEDIA_DURATION) == 1791
|
|
assert state.attributes.get(ATTR_MEDIA_POSITION) == 263
|
|
assert state.attributes.get(ATTR_MEDIA_POSITION_UPDATED_AT)
|
|
assert state.attributes.get(ATTR_MEDIA_TITLE) == "Tyler's Ultimate"
|
|
assert state.attributes.get(ATTR_MEDIA_SERIES_TITLE) == "Spaghetti and Clam Sauce"
|
|
assert state.attributes.get(ATTR_MEDIA_CHANNEL) == "{} ({})".format("FOODHD", "231")
|
|
assert state.attributes.get(ATTR_INPUT_SOURCE) == "231"
|
|
assert not state.attributes.get(ATTR_MEDIA_CURRENTLY_RECORDING)
|
|
assert state.attributes.get(ATTR_MEDIA_RATING) == "No Rating"
|
|
assert state.attributes.get(ATTR_MEDIA_RECORDED)
|
|
assert state.attributes.get(ATTR_MEDIA_START_TIME) == datetime(
|
|
2010, 7, 5, 15, 0, 8, tzinfo=dt_util.UTC
|
|
)
|
|
|
|
state = hass.states.get(MUSIC_ENTITY_ID)
|
|
assert state.state == STATE_PLAYING
|
|
|
|
assert state.attributes.get(ATTR_MEDIA_CONTENT_ID) == "76917562"
|
|
assert state.attributes.get(ATTR_MEDIA_CONTENT_TYPE) == MediaType.MUSIC
|
|
assert state.attributes.get(ATTR_MEDIA_DURATION) == 86400
|
|
assert state.attributes.get(ATTR_MEDIA_POSITION) == 15050
|
|
assert state.attributes.get(ATTR_MEDIA_POSITION_UPDATED_AT)
|
|
assert state.attributes.get(ATTR_MEDIA_TITLE) == "Sparkle In Your Eyes"
|
|
assert state.attributes.get(ATTR_MEDIA_ARTIST) == "Gerald Albright"
|
|
assert state.attributes.get(ATTR_MEDIA_ALBUM_NAME) == "Slam Dunk (2014)"
|
|
assert state.attributes.get(ATTR_MEDIA_SERIES_TITLE) is None
|
|
assert state.attributes.get(ATTR_MEDIA_CHANNEL) == "{} ({})".format("MCSJ", "851")
|
|
assert state.attributes.get(ATTR_INPUT_SOURCE) == "851"
|
|
assert not state.attributes.get(ATTR_MEDIA_CURRENTLY_RECORDING)
|
|
assert state.attributes.get(ATTR_MEDIA_RATING) == "TV-PG"
|
|
assert not state.attributes.get(ATTR_MEDIA_RECORDED)
|
|
assert state.attributes.get(ATTR_MEDIA_START_TIME) == datetime(
|
|
2020, 3, 21, 10, 0, 0, tzinfo=dt_util.UTC
|
|
)
|
|
|
|
state = hass.states.get(STANDBY_ENTITY_ID)
|
|
assert state.state == STATE_OFF
|
|
|
|
assert state.attributes.get(ATTR_MEDIA_CONTENT_ID) is None
|
|
assert state.attributes.get(ATTR_MEDIA_CONTENT_TYPE) is None
|
|
assert state.attributes.get(ATTR_MEDIA_DURATION) is None
|
|
assert state.attributes.get(ATTR_MEDIA_POSITION) is None
|
|
assert state.attributes.get(ATTR_MEDIA_POSITION_UPDATED_AT) is None
|
|
assert state.attributes.get(ATTR_MEDIA_TITLE) is None
|
|
assert state.attributes.get(ATTR_MEDIA_ARTIST) is None
|
|
assert state.attributes.get(ATTR_MEDIA_ALBUM_NAME) is None
|
|
assert state.attributes.get(ATTR_MEDIA_SERIES_TITLE) is None
|
|
assert state.attributes.get(ATTR_MEDIA_CHANNEL) is None
|
|
assert state.attributes.get(ATTR_INPUT_SOURCE) is None
|
|
assert not state.attributes.get(ATTR_MEDIA_CURRENTLY_RECORDING)
|
|
assert state.attributes.get(ATTR_MEDIA_RATING) is None
|
|
assert not state.attributes.get(ATTR_MEDIA_RECORDED)
|
|
|
|
state = hass.states.get(RESTRICTED_ENTITY_ID)
|
|
assert state.state == STATE_PLAYING
|
|
|
|
assert state.attributes.get(ATTR_MEDIA_CONTENT_ID) is None
|
|
assert state.attributes.get(ATTR_MEDIA_CONTENT_TYPE) is None
|
|
assert state.attributes.get(ATTR_MEDIA_DURATION) is None
|
|
assert state.attributes.get(ATTR_MEDIA_POSITION) is None
|
|
assert state.attributes.get(ATTR_MEDIA_POSITION_UPDATED_AT) is None
|
|
assert state.attributes.get(ATTR_MEDIA_TITLE) is None
|
|
assert state.attributes.get(ATTR_MEDIA_ARTIST) is None
|
|
assert state.attributes.get(ATTR_MEDIA_ALBUM_NAME) is None
|
|
assert state.attributes.get(ATTR_MEDIA_SERIES_TITLE) is None
|
|
assert state.attributes.get(ATTR_MEDIA_CHANNEL) is None
|
|
assert state.attributes.get(ATTR_INPUT_SOURCE) is None
|
|
assert not state.attributes.get(ATTR_MEDIA_CURRENTLY_RECORDING)
|
|
assert state.attributes.get(ATTR_MEDIA_RATING) is None
|
|
assert not state.attributes.get(ATTR_MEDIA_RECORDED)
|
|
|
|
state = hass.states.get(UNAVAILABLE_ENTITY_ID)
|
|
assert state.state == STATE_UNAVAILABLE
|
|
|
|
|
|
async def test_attributes_paused(
|
|
hass: HomeAssistant,
|
|
mock_now: dt_util.dt.datetime,
|
|
aioclient_mock: AiohttpClientMocker,
|
|
) -> None:
|
|
"""Test attributes while paused."""
|
|
await setup_integration(hass, aioclient_mock)
|
|
|
|
state = hass.states.get(CLIENT_ENTITY_ID)
|
|
last_updated = state.attributes.get(ATTR_MEDIA_POSITION_UPDATED_AT)
|
|
|
|
# Test to make sure that ATTR_MEDIA_POSITION_UPDATED_AT is not
|
|
# updated if TV is paused.
|
|
with patch(
|
|
"homeassistant.util.dt.utcnow", return_value=mock_now + timedelta(minutes=5)
|
|
):
|
|
await async_media_pause(hass, CLIENT_ENTITY_ID)
|
|
await hass.async_block_till_done()
|
|
|
|
state = hass.states.get(CLIENT_ENTITY_ID)
|
|
assert state.state == STATE_PAUSED
|
|
assert state.attributes.get(ATTR_MEDIA_POSITION_UPDATED_AT) == last_updated
|
|
|
|
|
|
async def test_main_services(
|
|
hass: HomeAssistant,
|
|
mock_now: dt_util.dt.datetime,
|
|
aioclient_mock: AiohttpClientMocker,
|
|
) -> None:
|
|
"""Test the different services."""
|
|
await setup_integration(hass, aioclient_mock)
|
|
|
|
with patch("directv.DIRECTV.remote") as remote_mock:
|
|
await async_turn_off(hass, MAIN_ENTITY_ID)
|
|
await hass.async_block_till_done()
|
|
remote_mock.assert_called_once_with("poweroff", "0")
|
|
|
|
with patch("directv.DIRECTV.remote") as remote_mock:
|
|
await async_turn_on(hass, MAIN_ENTITY_ID)
|
|
await hass.async_block_till_done()
|
|
remote_mock.assert_called_once_with("poweron", "0")
|
|
|
|
with patch("directv.DIRECTV.remote") as remote_mock:
|
|
await async_media_pause(hass, MAIN_ENTITY_ID)
|
|
await hass.async_block_till_done()
|
|
remote_mock.assert_called_once_with("pause", "0")
|
|
|
|
with patch("directv.DIRECTV.remote") as remote_mock:
|
|
await async_media_play(hass, MAIN_ENTITY_ID)
|
|
await hass.async_block_till_done()
|
|
remote_mock.assert_called_once_with("play", "0")
|
|
|
|
with patch("directv.DIRECTV.remote") as remote_mock:
|
|
await async_media_next_track(hass, MAIN_ENTITY_ID)
|
|
await hass.async_block_till_done()
|
|
remote_mock.assert_called_once_with("ffwd", "0")
|
|
|
|
with patch("directv.DIRECTV.remote") as remote_mock:
|
|
await async_media_previous_track(hass, MAIN_ENTITY_ID)
|
|
await hass.async_block_till_done()
|
|
remote_mock.assert_called_once_with("rew", "0")
|
|
|
|
with patch("directv.DIRECTV.remote") as remote_mock:
|
|
await async_media_stop(hass, MAIN_ENTITY_ID)
|
|
await hass.async_block_till_done()
|
|
remote_mock.assert_called_once_with("stop", "0")
|
|
|
|
with patch("directv.DIRECTV.tune") as tune_mock:
|
|
await async_play_media(hass, "channel", 312, MAIN_ENTITY_ID)
|
|
await hass.async_block_till_done()
|
|
tune_mock.assert_called_once_with("312", "0")
|