"""Tests for the todo integration.""" from collections.abc import Generator import datetime from typing import Any from unittest.mock import AsyncMock import zoneinfo import pytest import voluptuous as vol from homeassistant.components import conversation from homeassistant.components.homeassistant.exposed_entities import async_expose_entity from homeassistant.components.todo import ( ATTR_DESCRIPTION, ATTR_DUE_DATE, ATTR_DUE_DATETIME, ATTR_ITEM, ATTR_RENAME, ATTR_STATUS, DOMAIN, TodoItem, TodoItemStatus, TodoListEntity, TodoListEntityFeature, TodoServices, intent as todo_intent, ) from homeassistant.config_entries import ConfigEntry, ConfigEntryState, ConfigFlow from homeassistant.const import ATTR_ENTITY_ID, ATTR_SUPPORTED_FEATURES, Platform from homeassistant.core import HomeAssistant from homeassistant.exceptions import HomeAssistantError, ServiceValidationError from homeassistant.helpers import intent from homeassistant.helpers.entity_platform import AddEntitiesCallback from homeassistant.setup import async_setup_component from tests.common import ( MockConfigEntry, MockModule, MockPlatform, mock_config_flow, mock_integration, mock_platform, ) from tests.typing import WebSocketGenerator TEST_DOMAIN = "test" ITEM_1 = { "uid": "1", "summary": "Item #1", "status": "needs_action", } ITEM_2 = { "uid": "2", "summary": "Item #2", "status": "completed", } TEST_TIMEZONE = zoneinfo.ZoneInfo("America/Regina") TEST_OFFSET = "-06:00" class MockFlow(ConfigFlow): """Test flow.""" class MockTodoListEntity(TodoListEntity): """Test todo list entity.""" def __init__(self, items: list[TodoItem] | None = None) -> None: """Initialize entity.""" self._attr_todo_items = items or [] @property def items(self) -> list[TodoItem]: """Return the items in the To-do list.""" return self._attr_todo_items async def async_create_todo_item(self, item: TodoItem) -> None: """Add an item to the To-do list.""" self._attr_todo_items.append(item) async def async_delete_todo_items(self, uids: list[str]) -> None: """Delete an item in the To-do list.""" self._attr_todo_items = [item for item in self.items if item.uid not in uids] @pytest.fixture(autouse=True) def config_flow_fixture(hass: HomeAssistant) -> Generator[None]: """Mock config flow.""" mock_platform(hass, f"{TEST_DOMAIN}.config_flow") with mock_config_flow(TEST_DOMAIN, MockFlow): yield @pytest.fixture(autouse=True) def mock_setup_integration(hass: HomeAssistant) -> None: """Fixture to set up a mock integration.""" async def async_setup_entry_init( hass: HomeAssistant, config_entry: ConfigEntry ) -> bool: """Set up test config entry.""" await hass.config_entries.async_forward_entry_setups(config_entry, [DOMAIN]) return True async def async_unload_entry_init( hass: HomeAssistant, config_entry: ConfigEntry, ) -> bool: await hass.config_entries.async_unload_platforms(config_entry, [Platform.TODO]) return True mock_platform(hass, f"{TEST_DOMAIN}.config_flow") mock_integration( hass, MockModule( TEST_DOMAIN, async_setup_entry=async_setup_entry_init, async_unload_entry=async_unload_entry_init, ), ) @pytest.fixture(autouse=True) async def set_time_zone(hass: HomeAssistant) -> None: """Set the time zone for the tests that keesp UTC-6 all year round.""" await hass.config.async_set_time_zone("America/Regina") async def create_mock_platform( hass: HomeAssistant, entities: list[TodoListEntity], ) -> MockConfigEntry: """Create a todo platform with the specified entities.""" async def async_setup_entry_platform( hass: HomeAssistant, config_entry: ConfigEntry, async_add_entities: AddEntitiesCallback, ) -> None: """Set up test event platform via config entry.""" async_add_entities(entities) mock_platform( hass, f"{TEST_DOMAIN}.{DOMAIN}", MockPlatform(async_setup_entry=async_setup_entry_platform), ) config_entry = MockConfigEntry(domain=TEST_DOMAIN) config_entry.add_to_hass(hass) assert await hass.config_entries.async_setup(config_entry.entry_id) await hass.async_block_till_done() return config_entry @pytest.fixture(name="test_entity_items") def mock_test_entity_items() -> list[TodoItem]: """Fixture that creates the items returned by the test entity.""" return [ TodoItem(summary="Item #1", uid="1", status=TodoItemStatus.NEEDS_ACTION), TodoItem(summary="Item #2", uid="2", status=TodoItemStatus.COMPLETED), ] @pytest.fixture(name="test_entity") def mock_test_entity(test_entity_items: list[TodoItem]) -> TodoListEntity: """Fixture that creates a test TodoList entity with mock service calls.""" entity1 = MockTodoListEntity(test_entity_items) entity1.entity_id = "todo.entity1" entity1._attr_supported_features = ( TodoListEntityFeature.CREATE_TODO_ITEM | TodoListEntityFeature.UPDATE_TODO_ITEM | TodoListEntityFeature.DELETE_TODO_ITEM | TodoListEntityFeature.MOVE_TODO_ITEM ) entity1.async_create_todo_item = AsyncMock(wraps=entity1.async_create_todo_item) entity1.async_update_todo_item = AsyncMock() entity1.async_delete_todo_items = AsyncMock(wraps=entity1.async_delete_todo_items) entity1.async_move_todo_item = AsyncMock() return entity1 async def test_unload_entry( hass: HomeAssistant, test_entity: TodoListEntity, ) -> None: """Test unloading a config entry with a todo entity.""" config_entry = await create_mock_platform(hass, [test_entity]) assert config_entry.state is ConfigEntryState.LOADED state = hass.states.get("todo.entity1") assert state assert await hass.config_entries.async_unload(config_entry.entry_id) await hass.async_block_till_done() assert config_entry.state is ConfigEntryState.NOT_LOADED state = hass.states.get("todo.entity1") assert not state async def test_list_todo_items( hass: HomeAssistant, hass_ws_client: WebSocketGenerator, test_entity: TodoListEntity, ) -> None: """Test listing items in a To-do list.""" await create_mock_platform(hass, [test_entity]) state = hass.states.get("todo.entity1") assert state assert state.state == "1" assert state.attributes == {"supported_features": 15} client = await hass_ws_client(hass) await client.send_json( {"id": 1, "type": "todo/item/list", "entity_id": "todo.entity1"} ) resp = await client.receive_json() assert resp.get("id") == 1 assert resp.get("success") assert resp.get("result") == { "items": [ ITEM_1, ITEM_2, ] } @pytest.mark.parametrize( ("service_data", "expected_items"), [ ({}, [ITEM_1, ITEM_2]), ( {ATTR_STATUS: [TodoItemStatus.COMPLETED, TodoItemStatus.NEEDS_ACTION]}, [ITEM_1, ITEM_2], ), ({ATTR_STATUS: [TodoItemStatus.NEEDS_ACTION]}, [ITEM_1]), ({ATTR_STATUS: [TodoItemStatus.COMPLETED]}, [ITEM_2]), ], ) async def test_get_items_service( hass: HomeAssistant, hass_ws_client: WebSocketGenerator, test_entity: TodoListEntity, service_data: dict[str, Any], expected_items: list[dict[str, Any]], ) -> None: """Test listing items in a To-do list from a service call.""" await create_mock_platform(hass, [test_entity]) state = hass.states.get("todo.entity1") assert state assert state.state == "1" assert state.attributes == {ATTR_SUPPORTED_FEATURES: 15} result = await hass.services.async_call( DOMAIN, TodoServices.GET_ITEMS, service_data, target={ATTR_ENTITY_ID: "todo.entity1"}, blocking=True, return_response=True, ) assert result == {"todo.entity1": {"items": expected_items}} async def test_unsupported_websocket( hass: HomeAssistant, hass_ws_client: WebSocketGenerator, ) -> None: """Test a To-do list for an entity that does not exist.""" entity1 = TodoListEntity() entity1.entity_id = "todo.entity1" await create_mock_platform(hass, [entity1]) client = await hass_ws_client(hass) await client.send_json( { "id": 1, "type": "todo/item/list", "entity_id": "todo.unknown", } ) resp = await client.receive_json() assert resp.get("id") == 1 assert resp.get("error", {}).get("code") == "not_found" async def test_add_item_service( hass: HomeAssistant, test_entity: TodoListEntity, ) -> None: """Test adding an item in a To-do list.""" await create_mock_platform(hass, [test_entity]) await hass.services.async_call( DOMAIN, TodoServices.ADD_ITEM, {ATTR_ITEM: "New item"}, target={ATTR_ENTITY_ID: "todo.entity1"}, blocking=True, ) args = test_entity.async_create_todo_item.call_args assert args item = args.kwargs.get("item") assert item assert item.uid is None assert item.summary == "New item" assert item.status == TodoItemStatus.NEEDS_ACTION async def test_add_item_service_raises( hass: HomeAssistant, test_entity: TodoListEntity, ) -> None: """Test adding an item in a To-do list that raises an error.""" await create_mock_platform(hass, [test_entity]) test_entity.async_create_todo_item.side_effect = HomeAssistantError("Ooops") with pytest.raises(HomeAssistantError, match="Ooops"): await hass.services.async_call( DOMAIN, TodoServices.ADD_ITEM, {ATTR_ITEM: "New item"}, target={ATTR_ENTITY_ID: "todo.entity1"}, blocking=True, ) @pytest.mark.parametrize( ("item_data", "expected_exception", "expected_error"), [ ({}, vol.Invalid, "required key not provided"), ({ATTR_ITEM: ""}, vol.Invalid, "length of value must be at least 1"), ( {ATTR_ITEM: "Submit forms", ATTR_DESCRIPTION: "Submit tax forms"}, ServiceValidationError, "does not support setting field: description", ), ( {ATTR_ITEM: "Submit forms", ATTR_DUE_DATE: "2023-11-17"}, ServiceValidationError, "does not support setting field: due_date", ), ( { ATTR_ITEM: "Submit forms", ATTR_DUE_DATETIME: f"2023-11-17T17:00:00{TEST_OFFSET}", }, ServiceValidationError, "does not support setting field: due_datetime", ), ], ) async def test_add_item_service_invalid_input( hass: HomeAssistant, test_entity: TodoListEntity, item_data: dict[str, Any], expected_exception: str, expected_error: str, ) -> None: """Test invalid input to the add item service.""" await create_mock_platform(hass, [test_entity]) with pytest.raises(expected_exception) as exc: await hass.services.async_call( DOMAIN, TodoServices.ADD_ITEM, item_data, target={ATTR_ENTITY_ID: "todo.entity1"}, blocking=True, ) assert expected_error in str(exc.value) @pytest.mark.parametrize( ("supported_entity_feature", "item_data", "expected_item"), [ ( TodoListEntityFeature.SET_DUE_DATE_ON_ITEM, {ATTR_ITEM: "New item", ATTR_DUE_DATE: "2023-11-13"}, TodoItem( summary="New item", status=TodoItemStatus.NEEDS_ACTION, due=datetime.date(2023, 11, 13), ), ), ( TodoListEntityFeature.SET_DUE_DATETIME_ON_ITEM, { ATTR_ITEM: "New item", ATTR_DUE_DATETIME: f"2023-11-13T17:00:00{TEST_OFFSET}", }, TodoItem( summary="New item", status=TodoItemStatus.NEEDS_ACTION, due=datetime.datetime(2023, 11, 13, 17, 00, 00, tzinfo=TEST_TIMEZONE), ), ), ( TodoListEntityFeature.SET_DUE_DATETIME_ON_ITEM, {ATTR_ITEM: "New item", ATTR_DUE_DATETIME: "2023-11-13T17:00:00+00:00"}, TodoItem( summary="New item", status=TodoItemStatus.NEEDS_ACTION, due=datetime.datetime(2023, 11, 13, 11, 00, 00, tzinfo=TEST_TIMEZONE), ), ), ( TodoListEntityFeature.SET_DUE_DATETIME_ON_ITEM, {ATTR_ITEM: "New item", ATTR_DUE_DATETIME: "2023-11-13"}, TodoItem( summary="New item", status=TodoItemStatus.NEEDS_ACTION, due=datetime.datetime(2023, 11, 13, 0, 00, 00, tzinfo=TEST_TIMEZONE), ), ), ( TodoListEntityFeature.SET_DESCRIPTION_ON_ITEM, {ATTR_ITEM: "New item", ATTR_DESCRIPTION: "Submit revised draft"}, TodoItem( summary="New item", status=TodoItemStatus.NEEDS_ACTION, description="Submit revised draft", ), ), ], ) async def test_add_item_service_extended_fields( hass: HomeAssistant, test_entity: TodoListEntity, supported_entity_feature: int, item_data: dict[str, Any], expected_item: TodoItem, ) -> None: """Test adding an item in a To-do list.""" test_entity._attr_supported_features |= supported_entity_feature await create_mock_platform(hass, [test_entity]) await hass.services.async_call( DOMAIN, TodoServices.ADD_ITEM, {ATTR_ITEM: "New item", **item_data}, target={ATTR_ENTITY_ID: "todo.entity1"}, blocking=True, ) args = test_entity.async_create_todo_item.call_args assert args item = args.kwargs.get("item") assert item == expected_item async def test_update_todo_item_service_by_id( hass: HomeAssistant, test_entity: TodoListEntity, ) -> None: """Test updating an item in a To-do list.""" await create_mock_platform(hass, [test_entity]) await hass.services.async_call( DOMAIN, TodoServices.UPDATE_ITEM, {ATTR_ITEM: "1", ATTR_RENAME: "Updated item", ATTR_STATUS: "completed"}, target={ATTR_ENTITY_ID: "todo.entity1"}, blocking=True, ) args = test_entity.async_update_todo_item.call_args assert args item = args.kwargs.get("item") assert item assert item.uid == "1" assert item.summary == "Updated item" assert item.status == TodoItemStatus.COMPLETED async def test_update_todo_item_service_by_id_status_only( hass: HomeAssistant, test_entity: TodoListEntity, ) -> None: """Test updating an item in a To-do list.""" await create_mock_platform(hass, [test_entity]) await hass.services.async_call( DOMAIN, TodoServices.UPDATE_ITEM, {ATTR_ITEM: "1", ATTR_STATUS: "completed"}, target={ATTR_ENTITY_ID: "todo.entity1"}, blocking=True, ) args = test_entity.async_update_todo_item.call_args assert args item = args.kwargs.get("item") assert item assert item.uid == "1" assert item.summary == "Item #1" assert item.status == TodoItemStatus.COMPLETED async def test_update_todo_item_service_by_id_rename( hass: HomeAssistant, test_entity: TodoListEntity, ) -> None: """Test updating an item in a To-do list.""" await create_mock_platform(hass, [test_entity]) await hass.services.async_call( DOMAIN, TodoServices.UPDATE_ITEM, {ATTR_ITEM: "1", "rename": "Updated item"}, target={ATTR_ENTITY_ID: "todo.entity1"}, blocking=True, ) args = test_entity.async_update_todo_item.call_args assert args item = args.kwargs.get("item") assert item assert item.uid == "1" assert item.summary == "Updated item" assert item.status == TodoItemStatus.NEEDS_ACTION async def test_update_todo_item_service_raises( hass: HomeAssistant, test_entity: TodoListEntity, ) -> None: """Test updating an item in a To-do list that raises an error.""" await create_mock_platform(hass, [test_entity]) await hass.services.async_call( DOMAIN, TodoServices.UPDATE_ITEM, {ATTR_ITEM: "1", "rename": "Updated item", "status": "completed"}, target={ATTR_ENTITY_ID: "todo.entity1"}, blocking=True, ) test_entity.async_update_todo_item.side_effect = HomeAssistantError("Ooops") with pytest.raises(HomeAssistantError, match="Ooops"): await hass.services.async_call( DOMAIN, TodoServices.UPDATE_ITEM, {ATTR_ITEM: "1", "rename": "Updated item", "status": "completed"}, target={ATTR_ENTITY_ID: "todo.entity1"}, blocking=True, ) async def test_update_todo_item_service_by_summary( hass: HomeAssistant, test_entity: TodoListEntity, ) -> None: """Test updating an item in a To-do list by summary.""" await create_mock_platform(hass, [test_entity]) await hass.services.async_call( DOMAIN, TodoServices.UPDATE_ITEM, {ATTR_ITEM: "Item #1", "rename": "Something else", "status": "completed"}, target={ATTR_ENTITY_ID: "todo.entity1"}, blocking=True, ) args = test_entity.async_update_todo_item.call_args assert args item = args.kwargs.get("item") assert item assert item.uid == "1" assert item.summary == "Something else" assert item.status == TodoItemStatus.COMPLETED async def test_update_todo_item_service_by_summary_only_status( hass: HomeAssistant, test_entity: TodoListEntity, ) -> None: """Test updating an item in a To-do list by summary.""" await create_mock_platform(hass, [test_entity]) await hass.services.async_call( DOMAIN, TodoServices.UPDATE_ITEM, {ATTR_ITEM: "Item #1", "rename": "Something else"}, target={ATTR_ENTITY_ID: "todo.entity1"}, blocking=True, ) args = test_entity.async_update_todo_item.call_args assert args item = args.kwargs.get("item") assert item assert item.uid == "1" assert item.summary == "Something else" assert item.status == TodoItemStatus.NEEDS_ACTION async def test_update_todo_item_service_by_summary_not_found( hass: HomeAssistant, test_entity: TodoListEntity, ) -> None: """Test updating an item in a To-do list by summary which is not found.""" await create_mock_platform(hass, [test_entity]) with pytest.raises(ServiceValidationError, match="Unable to find"): await hass.services.async_call( DOMAIN, TodoServices.UPDATE_ITEM, {ATTR_ITEM: "Item #7", "status": "completed"}, target={ATTR_ENTITY_ID: "todo.entity1"}, blocking=True, ) @pytest.mark.parametrize( ("item_data", "expected_error"), [ ({}, r"required key not provided @ data\['item'\]"), ({"status": "needs_action"}, r"required key not provided @ data\['item'\]"), ({"item": "Item #1"}, "must contain at least one of"), ( {"item": "", "status": "needs_action"}, "length of value must be at least 1", ), ], ) async def test_update_item_service_invalid_input( hass: HomeAssistant, test_entity: TodoListEntity, item_data: dict[str, Any], expected_error: str, ) -> None: """Test invalid input to the update item service.""" await create_mock_platform(hass, [test_entity]) with pytest.raises(vol.Invalid, match=expected_error): await hass.services.async_call( DOMAIN, "update_item", item_data, target={ATTR_ENTITY_ID: "todo.entity1"}, blocking=True, ) @pytest.mark.parametrize( ("update_data"), [ ({"due_datetime": f"2023-11-13T17:00:00{TEST_OFFSET}"}), ({"due_date": "2023-11-13"}), ({"description": "Submit revised draft"}), ], ) async def test_update_todo_item_field_unsupported( hass: HomeAssistant, test_entity: TodoListEntity, update_data: dict[str, Any], ) -> None: """Test updating an item in a To-do list.""" await create_mock_platform(hass, [test_entity]) with pytest.raises(ServiceValidationError, match="does not support"): await hass.services.async_call( DOMAIN, TodoServices.UPDATE_ITEM, {ATTR_ITEM: "1", **update_data}, target={ATTR_ENTITY_ID: "todo.entity1"}, blocking=True, ) @pytest.mark.parametrize( ("supported_entity_feature", "update_data", "expected_update"), [ ( TodoListEntityFeature.SET_DUE_DATE_ON_ITEM, {"due_date": "2023-11-13"}, TodoItem( uid="1", summary="Item #1", status=TodoItemStatus.NEEDS_ACTION, due=datetime.date(2023, 11, 13), ), ), ( TodoListEntityFeature.SET_DUE_DATETIME_ON_ITEM, {"due_datetime": f"2023-11-13T17:00:00{TEST_OFFSET}"}, TodoItem( uid="1", summary="Item #1", status=TodoItemStatus.NEEDS_ACTION, due=datetime.datetime(2023, 11, 13, 17, 0, 0, tzinfo=TEST_TIMEZONE), ), ), ( TodoListEntityFeature.SET_DESCRIPTION_ON_ITEM, {"description": "Submit revised draft"}, TodoItem( uid="1", summary="Item #1", status=TodoItemStatus.NEEDS_ACTION, description="Submit revised draft", ), ), ], ) async def test_update_todo_item_extended_fields( hass: HomeAssistant, test_entity: TodoListEntity, supported_entity_feature: int, update_data: dict[str, Any], expected_update: TodoItem, ) -> None: """Test updating an item in a To-do list.""" test_entity._attr_supported_features |= supported_entity_feature await create_mock_platform(hass, [test_entity]) await hass.services.async_call( DOMAIN, TodoServices.UPDATE_ITEM, {ATTR_ITEM: "1", **update_data}, target={ATTR_ENTITY_ID: "todo.entity1"}, blocking=True, ) args = test_entity.async_update_todo_item.call_args assert args item = args.kwargs.get("item") assert item == expected_update @pytest.mark.parametrize( ("test_entity_items", "update_data", "expected_update"), [ ( [TodoItem(uid="1", summary="Summary", description="description")], {"description": "Submit revised draft"}, TodoItem(uid="1", summary="Summary", description="Submit revised draft"), ), ( [TodoItem(uid="1", summary="Summary", description="description")], {"description": ""}, TodoItem(uid="1", summary="Summary", description=""), ), ( [TodoItem(uid="1", summary="Summary", description="description")], {"description": None}, TodoItem(uid="1", summary="Summary"), ), ( [TodoItem(uid="1", summary="Summary", due=datetime.date(2024, 1, 1))], {"due_date": datetime.date(2024, 1, 2)}, TodoItem(uid="1", summary="Summary", due=datetime.date(2024, 1, 2)), ), ( [TodoItem(uid="1", summary="Summary", due=datetime.date(2024, 1, 1))], {"due_date": None}, TodoItem(uid="1", summary="Summary"), ), ( [TodoItem(uid="1", summary="Summary", due=datetime.date(2024, 1, 1))], {"due_datetime": datetime.datetime(2024, 1, 1, 10, 0, 0)}, TodoItem( uid="1", summary="Summary", due=datetime.datetime( 2024, 1, 1, 10, 0, 0, tzinfo=zoneinfo.ZoneInfo(key="America/Regina") ), ), ), ( [ TodoItem( uid="1", summary="Summary", due=datetime.datetime(2024, 1, 1, 10, 0, 0), ) ], {"due_datetime": None}, TodoItem(uid="1", summary="Summary"), ), ], ids=[ "overwrite_description", "overwrite_empty_description", "clear_description", "overwrite_due_date", "clear_due_date", "overwrite_due_date_with_time", "clear_due_date_time", ], ) async def test_update_todo_item_extended_fields_overwrite_existing_values( hass: HomeAssistant, test_entity: TodoListEntity, update_data: dict[str, Any], expected_update: TodoItem, ) -> None: """Test updating an item in a To-do list.""" test_entity._attr_supported_features |= ( TodoListEntityFeature.SET_DESCRIPTION_ON_ITEM | TodoListEntityFeature.SET_DUE_DATE_ON_ITEM | TodoListEntityFeature.SET_DUE_DATETIME_ON_ITEM ) await create_mock_platform(hass, [test_entity]) await hass.services.async_call( DOMAIN, TodoServices.UPDATE_ITEM, {ATTR_ITEM: "1", **update_data}, target={ATTR_ENTITY_ID: "todo.entity1"}, blocking=True, ) args = test_entity.async_update_todo_item.call_args assert args item = args.kwargs.get("item") assert item == expected_update async def test_remove_todo_item_service_by_id( hass: HomeAssistant, test_entity: TodoListEntity, ) -> None: """Test removing an item in a To-do list.""" await create_mock_platform(hass, [test_entity]) await hass.services.async_call( DOMAIN, TodoServices.REMOVE_ITEM, {ATTR_ITEM: ["1", "2"]}, target={ATTR_ENTITY_ID: "todo.entity1"}, blocking=True, ) args = test_entity.async_delete_todo_items.call_args assert args assert args.kwargs.get("uids") == ["1", "2"] async def test_remove_todo_item_service_raises( hass: HomeAssistant, test_entity: TodoListEntity, ) -> None: """Test removing an item in a To-do list that raises an error.""" await create_mock_platform(hass, [test_entity]) test_entity.async_delete_todo_items.side_effect = HomeAssistantError("Ooops") with pytest.raises(HomeAssistantError, match="Ooops"): await hass.services.async_call( DOMAIN, TodoServices.REMOVE_ITEM, {ATTR_ITEM: ["1", "2"]}, target={ATTR_ENTITY_ID: "todo.entity1"}, blocking=True, ) async def test_remove_todo_item_service_invalid_input( hass: HomeAssistant, test_entity: TodoListEntity, ) -> None: """Test invalid input to the remove item service.""" await create_mock_platform(hass, [test_entity]) with pytest.raises( vol.Invalid, match=r"required key not provided @ data\['item'\]" ): await hass.services.async_call( DOMAIN, TodoServices.REMOVE_ITEM, {}, target={ATTR_ENTITY_ID: "todo.entity1"}, blocking=True, ) async def test_remove_todo_item_service_by_summary( hass: HomeAssistant, test_entity: TodoListEntity, ) -> None: """Test removing an item in a To-do list by summary.""" await create_mock_platform(hass, [test_entity]) await hass.services.async_call( DOMAIN, TodoServices.REMOVE_ITEM, {ATTR_ITEM: ["Item #1"]}, target={ATTR_ENTITY_ID: "todo.entity1"}, blocking=True, ) args = test_entity.async_delete_todo_items.call_args assert args assert args.kwargs.get("uids") == ["1"] async def test_remove_todo_item_service_by_summary_not_found( hass: HomeAssistant, test_entity: TodoListEntity, ) -> None: """Test removing an item in a To-do list by summary which is not found.""" await create_mock_platform(hass, [test_entity]) with pytest.raises(ServiceValidationError, match="Unable to find"): await hass.services.async_call( DOMAIN, TodoServices.REMOVE_ITEM, {ATTR_ITEM: ["Item #7"]}, target={ATTR_ENTITY_ID: "todo.entity1"}, blocking=True, ) async def test_move_todo_item_service_by_id( hass: HomeAssistant, test_entity: TodoListEntity, hass_ws_client: WebSocketGenerator, ) -> None: """Test moving an item in a To-do list.""" await create_mock_platform(hass, [test_entity]) client = await hass_ws_client() await client.send_json( { "id": 1, "type": "todo/item/move", "entity_id": "todo.entity1", "uid": "item-1", "previous_uid": "item-2", } ) resp = await client.receive_json() assert resp.get("id") == 1 assert resp.get("success") args = test_entity.async_move_todo_item.call_args assert args assert args.kwargs.get("uid") == "item-1" assert args.kwargs.get("previous_uid") == "item-2" async def test_move_todo_item_service_raises( hass: HomeAssistant, test_entity: TodoListEntity, hass_ws_client: WebSocketGenerator, ) -> None: """Test moving an item in a To-do list that raises an error.""" await create_mock_platform(hass, [test_entity]) test_entity.async_move_todo_item.side_effect = HomeAssistantError("Ooops") client = await hass_ws_client() await client.send_json( { "id": 1, "type": "todo/item/move", "entity_id": "todo.entity1", "uid": "item-1", "previous_uid": "item-2", } ) resp = await client.receive_json() assert resp.get("id") == 1 assert resp.get("error", {}).get("code") == "failed" assert resp.get("error", {}).get("message") == "Ooops" @pytest.mark.parametrize( ("item_data", "expected_status", "expected_error"), [ ( {"entity_id": "todo.unknown", "uid": "item-1"}, "not_found", "Entity not found", ), ({"entity_id": "todo.entity1"}, "invalid_format", "required key not provided"), ( {"entity_id": "todo.entity1", "previous_uid": "item-2"}, "invalid_format", "required key not provided", ), ], ) async def test_move_todo_item_service_invalid_input( hass: HomeAssistant, test_entity: TodoListEntity, hass_ws_client: WebSocketGenerator, item_data: dict[str, Any], expected_status: str, expected_error: str, ) -> None: """Test invalid input for the move item service.""" await create_mock_platform(hass, [test_entity]) client = await hass_ws_client() await client.send_json( { "id": 1, "type": "todo/item/move", **item_data, } ) resp = await client.receive_json() assert resp.get("id") == 1 assert resp.get("error", {}).get("code") == expected_status assert expected_error in resp.get("error", {}).get("message") @pytest.mark.parametrize( ("service_name", "payload"), [ ( TodoServices.ADD_ITEM, { ATTR_ITEM: "New item", }, ), ( TodoServices.REMOVE_ITEM, { ATTR_ITEM: ["1"], }, ), ( TodoServices.UPDATE_ITEM, { ATTR_ITEM: "1", ATTR_RENAME: "Updated item", }, ), ( TodoServices.REMOVE_COMPLETED_ITEMS, None, ), ], ) async def test_unsupported_service( hass: HomeAssistant, service_name: str, payload: dict[str, Any] | None, ) -> None: """Test a To-do list that does not support features.""" entity1 = TodoListEntity() entity1.entity_id = "todo.entity1" await create_mock_platform(hass, [entity1]) with pytest.raises( HomeAssistantError, match="does not support this service", ): await hass.services.async_call( DOMAIN, service_name, payload, target={ATTR_ENTITY_ID: "todo.entity1"}, blocking=True, ) async def test_move_item_unsupported( hass: HomeAssistant, hass_ws_client: WebSocketGenerator, ) -> None: """Test invalid input for the move item service.""" entity1 = TodoListEntity() entity1.entity_id = "todo.entity1" await create_mock_platform(hass, [entity1]) client = await hass_ws_client() await client.send_json( { "id": 1, "type": "todo/item/move", "entity_id": "todo.entity1", "uid": "item-1", "previous_uid": "item-2", } ) resp = await client.receive_json() assert resp.get("id") == 1 assert resp.get("error", {}).get("code") == "not_supported" async def test_add_item_intent( hass: HomeAssistant, hass_ws_client: WebSocketGenerator, ) -> None: """Test adding items to lists using an intent.""" assert await async_setup_component(hass, "homeassistant", {}) await todo_intent.async_setup_intents(hass) entity1 = MockTodoListEntity() entity1._attr_name = "List 1" entity1.entity_id = "todo.list_1" entity2 = MockTodoListEntity() entity2._attr_name = "List 2" entity2.entity_id = "todo.list_2" await create_mock_platform(hass, [entity1, entity2]) # Add to first list response = await intent.async_handle( hass, "test", todo_intent.INTENT_LIST_ADD_ITEM, {ATTR_ITEM: {"value": "beer"}, "name": {"value": "list 1"}}, assistant=conversation.DOMAIN, ) assert response.response_type == intent.IntentResponseType.ACTION_DONE assert len(entity1.items) == 1 assert len(entity2.items) == 0 assert entity1.items[0].summary == "beer" assert entity1.items[0].status == TodoItemStatus.NEEDS_ACTION entity1.items.clear() # Add to second list response = await intent.async_handle( hass, "test", todo_intent.INTENT_LIST_ADD_ITEM, {ATTR_ITEM: {"value": "cheese"}, "name": {"value": "List 2"}}, assistant=conversation.DOMAIN, ) assert response.response_type == intent.IntentResponseType.ACTION_DONE assert len(entity1.items) == 0 assert len(entity2.items) == 1 assert entity2.items[0].summary == "cheese" assert entity2.items[0].status == TodoItemStatus.NEEDS_ACTION # List name is case insensitive response = await intent.async_handle( hass, "test", todo_intent.INTENT_LIST_ADD_ITEM, {ATTR_ITEM: {"value": "wine"}, "name": {"value": "lIST 2"}}, assistant=conversation.DOMAIN, ) assert response.response_type == intent.IntentResponseType.ACTION_DONE assert len(entity1.items) == 0 assert len(entity2.items) == 2 assert entity2.items[1].summary == "wine" assert entity2.items[1].status == TodoItemStatus.NEEDS_ACTION # Should fail if lists are not exposed async_expose_entity(hass, conversation.DOMAIN, entity1.entity_id, False) async_expose_entity(hass, conversation.DOMAIN, entity2.entity_id, False) with pytest.raises(intent.MatchFailedError) as err: await intent.async_handle( hass, "test", todo_intent.INTENT_LIST_ADD_ITEM, {"item": {"value": "cookies"}, "name": {"value": "list 1"}}, assistant=conversation.DOMAIN, ) assert err.value.result.no_match_reason == intent.MatchFailedReason.ASSISTANT # Missing list with pytest.raises(intent.MatchFailedError): await intent.async_handle( hass, "test", todo_intent.INTENT_LIST_ADD_ITEM, {"item": {"value": "wine"}, "name": {"value": "This list does not exist"}}, assistant=conversation.DOMAIN, ) # Fail with empty name/item with pytest.raises(intent.InvalidSlotInfo): await intent.async_handle( hass, "test", todo_intent.INTENT_LIST_ADD_ITEM, {"item": {"value": "wine"}, "name": {"value": ""}}, assistant=conversation.DOMAIN, ) with pytest.raises(intent.InvalidSlotInfo): await intent.async_handle( hass, "test", todo_intent.INTENT_LIST_ADD_ITEM, {"item": {"value": ""}, "name": {"value": "list 1"}}, assistant=conversation.DOMAIN, ) async def test_remove_completed_items_service( hass: HomeAssistant, test_entity: TodoListEntity, ) -> None: """Test remove completed todo items service.""" await create_mock_platform(hass, [test_entity]) await hass.services.async_call( DOMAIN, TodoServices.REMOVE_COMPLETED_ITEMS, target={ATTR_ENTITY_ID: "todo.entity1"}, blocking=True, ) args = test_entity.async_delete_todo_items.call_args assert args assert args.kwargs.get("uids") == ["2"] test_entity.async_delete_todo_items.reset_mock() # calling service multiple times will not call the entity method await hass.services.async_call( DOMAIN, TodoServices.REMOVE_COMPLETED_ITEMS, target={ATTR_ENTITY_ID: "todo.entity1"}, blocking=True, ) test_entity.async_delete_todo_items.assert_not_called() async def test_remove_completed_items_service_raises( hass: HomeAssistant, test_entity: TodoListEntity, ) -> None: """Test removing all completed item from a To-do list that raises an error.""" await create_mock_platform(hass, [test_entity]) test_entity.async_delete_todo_items.side_effect = HomeAssistantError("Ooops") with pytest.raises(HomeAssistantError, match="Ooops"): await hass.services.async_call( DOMAIN, TodoServices.REMOVE_COMPLETED_ITEMS, target={ATTR_ENTITY_ID: "todo.entity1"}, blocking=True, ) async def test_subscribe( hass: HomeAssistant, hass_ws_client: WebSocketGenerator, test_entity: TodoListEntity, ) -> None: """Test subscribing to todo updates.""" await create_mock_platform(hass, [test_entity]) client = await hass_ws_client(hass) await client.send_json_auto_id( { "type": "todo/item/subscribe", "entity_id": test_entity.entity_id, } ) msg = await client.receive_json() assert msg["success"] assert msg["result"] is None subscription_id = msg["id"] msg = await client.receive_json() assert msg["id"] == subscription_id assert msg["type"] == "event" event_message = msg["event"] assert event_message == { "items": [ { "summary": "Item #1", "uid": "1", "status": "needs_action", "due": None, "description": None, }, { "summary": "Item #2", "uid": "2", "status": "completed", "due": None, "description": None, }, ] } test_entity._attr_todo_items = [ *test_entity._attr_todo_items, TodoItem(summary="Item #3", uid="3", status=TodoItemStatus.NEEDS_ACTION), ] test_entity.async_write_ha_state() msg = await client.receive_json() event_message = msg["event"] assert event_message == { "items": [ { "summary": "Item #1", "uid": "1", "status": "needs_action", "due": None, "description": None, }, { "summary": "Item #2", "uid": "2", "status": "completed", "due": None, "description": None, }, { "summary": "Item #3", "uid": "3", "status": "needs_action", "due": None, "description": None, }, ] } test_entity._attr_todo_items = None test_entity.async_write_ha_state() msg = await client.receive_json() event_message = msg["event"] assert event_message == { "items": [], } async def test_subscribe_entity_does_not_exist( hass: HomeAssistant, hass_ws_client: WebSocketGenerator, test_entity: TodoListEntity, ) -> None: """Test failure to subscribe to an entity that does not exist.""" await create_mock_platform(hass, [test_entity]) client = await hass_ws_client(hass) await client.send_json_auto_id( { "type": "todo/item/subscribe", "entity_id": "todo.unknown", } ) msg = await client.receive_json() assert not msg["success"] assert msg["error"] == { "code": "invalid_entity_id", "message": "To-do list entity not found: todo.unknown", } @pytest.mark.parametrize( ("item_data", "expected_item_data"), [ ({"due": datetime.date(2023, 11, 17)}, {"due": "2023-11-17"}), ( {"due": datetime.datetime(2023, 11, 17, 17, 0, 0, tzinfo=TEST_TIMEZONE)}, {"due": f"2023-11-17T17:00:00{TEST_OFFSET}"}, ), ({"description": "Some description"}, {"description": "Some description"}), ], ) async def test_list_todo_items_extended_fields( hass: HomeAssistant, hass_ws_client: WebSocketGenerator, test_entity: TodoListEntity, item_data: dict[str, Any], expected_item_data: dict[str, Any], ) -> None: """Test listing items in a To-do list with extended fields.""" test_entity._attr_todo_items = [ TodoItem( **ITEM_1, **item_data, ), ] await create_mock_platform(hass, [test_entity]) client = await hass_ws_client(hass) await client.send_json( {"id": 1, "type": "todo/item/list", "entity_id": "todo.entity1"} ) resp = await client.receive_json() assert resp.get("id") == 1 assert resp.get("success") assert resp.get("result") == { "items": [ { **ITEM_1, **expected_item_data, }, ] } result = await hass.services.async_call( DOMAIN, "get_items", {}, target={ATTR_ENTITY_ID: "todo.entity1"}, blocking=True, return_response=True, ) assert result == { "todo.entity1": { "items": [ { **ITEM_1, **expected_item_data, }, ] } }