mirror of
https://github.com/home-assistant/core.git
synced 2025-07-21 12:17:07 +00:00
Add todo.remove_completed_items
service call (#104035)
* Extend `remove_item` service by status * update services.yaml * Create own service * add tests * Update tests/components/todo/test_init.py Co-authored-by: Martin Hjelmare <marhje52@gmail.com> --------- Co-authored-by: Martin Hjelmare <marhje52@gmail.com>
This commit is contained in:
parent
5527cbd78a
commit
9d3f374728
@ -90,6 +90,12 @@ async def async_setup(hass: HomeAssistant, config: ConfigType) -> bool:
|
|||||||
_async_get_todo_items,
|
_async_get_todo_items,
|
||||||
supports_response=SupportsResponse.ONLY,
|
supports_response=SupportsResponse.ONLY,
|
||||||
)
|
)
|
||||||
|
component.async_register_entity_service(
|
||||||
|
"remove_completed_items",
|
||||||
|
{},
|
||||||
|
_async_remove_completed_items,
|
||||||
|
required_features=[TodoListEntityFeature.DELETE_TODO_ITEM],
|
||||||
|
)
|
||||||
|
|
||||||
await component.async_setup(config)
|
await component.async_setup(config)
|
||||||
return True
|
return True
|
||||||
@ -284,3 +290,14 @@ async def _async_get_todo_items(
|
|||||||
if not (statuses := call.data.get("status")) or item.status in statuses
|
if not (statuses := call.data.get("status")) or item.status in statuses
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
async def _async_remove_completed_items(entity: TodoListEntity, _: ServiceCall) -> None:
|
||||||
|
"""Remove all completed items from the To-do list."""
|
||||||
|
uids = [
|
||||||
|
item.uid
|
||||||
|
for item in entity.todo_items or ()
|
||||||
|
if item.status == TodoItemStatus.COMPLETED and item.uid
|
||||||
|
]
|
||||||
|
if uids:
|
||||||
|
await entity.async_delete_todo_items(uids=uids)
|
||||||
|
@ -60,3 +60,5 @@ remove_item:
|
|||||||
required: true
|
required: true
|
||||||
selector:
|
selector:
|
||||||
text:
|
text:
|
||||||
|
|
||||||
|
remove_completed_items:
|
||||||
|
@ -44,6 +44,10 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"remove_completed_items": {
|
||||||
|
"name": "Remove all completed to-do list items",
|
||||||
|
"description": "Remove all to-do list items that have been completed."
|
||||||
|
},
|
||||||
"remove_item": {
|
"remove_item": {
|
||||||
"name": "Remove a to-do list item",
|
"name": "Remove a to-do list item",
|
||||||
"description": "Remove an existing to-do list item by its name.",
|
"description": "Remove an existing to-do list item by its name.",
|
||||||
|
@ -52,13 +52,22 @@ class MockFlow(ConfigFlow):
|
|||||||
class MockTodoListEntity(TodoListEntity):
|
class MockTodoListEntity(TodoListEntity):
|
||||||
"""Test todo list entity."""
|
"""Test todo list entity."""
|
||||||
|
|
||||||
def __init__(self) -> None:
|
def __init__(self, items: list[TodoItem] | None = None) -> None:
|
||||||
"""Initialize entity."""
|
"""Initialize entity."""
|
||||||
self.items: list[TodoItem] = []
|
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:
|
async def async_create_todo_item(self, item: TodoItem) -> None:
|
||||||
"""Add an item to the To-do list."""
|
"""Add an item to the To-do list."""
|
||||||
self.items.append(item)
|
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)
|
@pytest.fixture(autouse=True)
|
||||||
@ -130,7 +139,12 @@ async def create_mock_platform(
|
|||||||
@pytest.fixture(name="test_entity")
|
@pytest.fixture(name="test_entity")
|
||||||
def mock_test_entity() -> TodoListEntity:
|
def mock_test_entity() -> TodoListEntity:
|
||||||
"""Fixture that creates a test TodoList entity with mock service calls."""
|
"""Fixture that creates a test TodoList entity with mock service calls."""
|
||||||
entity1 = TodoListEntity()
|
entity1 = MockTodoListEntity(
|
||||||
|
[
|
||||||
|
TodoItem(summary="Item #1", uid="1", status=TodoItemStatus.NEEDS_ACTION),
|
||||||
|
TodoItem(summary="Item #2", uid="2", status=TodoItemStatus.COMPLETED),
|
||||||
|
]
|
||||||
|
)
|
||||||
entity1.entity_id = "todo.entity1"
|
entity1.entity_id = "todo.entity1"
|
||||||
entity1._attr_supported_features = (
|
entity1._attr_supported_features = (
|
||||||
TodoListEntityFeature.CREATE_TODO_ITEM
|
TodoListEntityFeature.CREATE_TODO_ITEM
|
||||||
@ -138,13 +152,9 @@ def mock_test_entity() -> TodoListEntity:
|
|||||||
| TodoListEntityFeature.DELETE_TODO_ITEM
|
| TodoListEntityFeature.DELETE_TODO_ITEM
|
||||||
| TodoListEntityFeature.MOVE_TODO_ITEM
|
| TodoListEntityFeature.MOVE_TODO_ITEM
|
||||||
)
|
)
|
||||||
entity1._attr_todo_items = [
|
entity1.async_create_todo_item = AsyncMock(wraps=entity1.async_create_todo_item)
|
||||||
TodoItem(summary="Item #1", uid="1", status=TodoItemStatus.NEEDS_ACTION),
|
|
||||||
TodoItem(summary="Item #2", uid="2", status=TodoItemStatus.COMPLETED),
|
|
||||||
]
|
|
||||||
entity1.async_create_todo_item = AsyncMock()
|
|
||||||
entity1.async_update_todo_item = AsyncMock()
|
entity1.async_update_todo_item = AsyncMock()
|
||||||
entity1.async_delete_todo_items = AsyncMock()
|
entity1.async_delete_todo_items = AsyncMock(wraps=entity1.async_delete_todo_items)
|
||||||
entity1.async_move_todo_item = AsyncMock()
|
entity1.async_move_todo_item = AsyncMock()
|
||||||
return entity1
|
return entity1
|
||||||
|
|
||||||
@ -763,12 +773,16 @@ async def test_move_todo_item_service_invalid_input(
|
|||||||
"rename": "Updated item",
|
"rename": "Updated item",
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
|
(
|
||||||
|
"remove_completed_items",
|
||||||
|
None,
|
||||||
|
),
|
||||||
],
|
],
|
||||||
)
|
)
|
||||||
async def test_unsupported_service(
|
async def test_unsupported_service(
|
||||||
hass: HomeAssistant,
|
hass: HomeAssistant,
|
||||||
service_name: str,
|
service_name: str,
|
||||||
payload: dict[str, Any],
|
payload: dict[str, Any] | None,
|
||||||
) -> None:
|
) -> None:
|
||||||
"""Test a To-do list that does not support features."""
|
"""Test a To-do list that does not support features."""
|
||||||
|
|
||||||
@ -879,3 +893,51 @@ async def test_add_item_intent(
|
|||||||
todo_intent.INTENT_LIST_ADD_ITEM,
|
todo_intent.INTENT_LIST_ADD_ITEM,
|
||||||
{"item": {"value": "wine"}, "name": {"value": "This list does not exist"}},
|
{"item": {"value": "wine"}, "name": {"value": "This list does not exist"}},
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
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,
|
||||||
|
"remove_completed_items",
|
||||||
|
target={"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,
|
||||||
|
"remove_completed_items",
|
||||||
|
target={"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,
|
||||||
|
"remove_completed_items",
|
||||||
|
target={"entity_id": "todo.entity1"},
|
||||||
|
blocking=True,
|
||||||
|
)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user