mirror of
https://github.com/home-assistant/core.git
synced 2025-07-25 06:07:17 +00:00
Remove duplicated options from input_select (#66680)
This commit is contained in:
parent
0138caa277
commit
eaf73318e1
@ -16,6 +16,7 @@ from homeassistant.const import (
|
||||
SERVICE_RELOAD,
|
||||
)
|
||||
from homeassistant.core import HomeAssistant, ServiceCall, callback
|
||||
from homeassistant.exceptions import HomeAssistantError
|
||||
from homeassistant.helpers import collection
|
||||
import homeassistant.helpers.config_validation as cv
|
||||
from homeassistant.helpers.entity_component import EntityComponent
|
||||
@ -42,6 +43,7 @@ SERVICE_SELECT_LAST = "select_last"
|
||||
SERVICE_SET_OPTIONS = "set_options"
|
||||
STORAGE_KEY = DOMAIN
|
||||
STORAGE_VERSION = 1
|
||||
STORAGE_VERSION_MINOR = 2
|
||||
|
||||
CREATE_FIELDS = {
|
||||
vol.Required(CONF_NAME): vol.All(str, vol.Length(min=1)),
|
||||
@ -57,6 +59,20 @@ UPDATE_FIELDS = {
|
||||
}
|
||||
|
||||
|
||||
def _remove_duplicates(options: list[str], name: str | None) -> list[str]:
|
||||
"""Remove duplicated options."""
|
||||
unique_options = list(dict.fromkeys(options))
|
||||
# This check was added in 2022.3
|
||||
# Reject YAML configured input_select with duplicates from 2022.6
|
||||
if len(unique_options) != len(options):
|
||||
_LOGGER.warning(
|
||||
"Input select '%s' with options %s had duplicated options, the duplicates have been removed",
|
||||
name or "<unnamed>",
|
||||
options,
|
||||
)
|
||||
return unique_options
|
||||
|
||||
|
||||
def _cv_input_select(cfg: dict[str, Any]) -> dict[str, Any]:
|
||||
"""Configure validation helper for input select (voluptuous)."""
|
||||
options = cfg[CONF_OPTIONS]
|
||||
@ -65,6 +81,7 @@ def _cv_input_select(cfg: dict[str, Any]) -> dict[str, Any]:
|
||||
raise vol.Invalid(
|
||||
f"initial state {initial} is not part of the options: {','.join(options)}"
|
||||
)
|
||||
cfg[CONF_OPTIONS] = _remove_duplicates(options, cfg.get(CONF_NAME))
|
||||
return cfg
|
||||
|
||||
|
||||
@ -89,6 +106,23 @@ CONFIG_SCHEMA = vol.Schema(
|
||||
RELOAD_SERVICE_SCHEMA = vol.Schema({})
|
||||
|
||||
|
||||
class InputSelectStore(Store):
|
||||
"""Store entity registry data."""
|
||||
|
||||
async def _async_migrate_func(
|
||||
self, old_major_version: int, old_minor_version: int, old_data: dict[str, Any]
|
||||
) -> dict[str, Any]:
|
||||
"""Migrate to the new version."""
|
||||
if old_major_version == 1:
|
||||
if old_minor_version < 2:
|
||||
for item in old_data["items"]:
|
||||
options = item[ATTR_OPTIONS]
|
||||
item[ATTR_OPTIONS] = _remove_duplicates(
|
||||
options, item.get(CONF_NAME)
|
||||
)
|
||||
return old_data
|
||||
|
||||
|
||||
async def async_setup(hass: HomeAssistant, config: ConfigType) -> bool:
|
||||
"""Set up an input select."""
|
||||
component = EntityComponent(_LOGGER, DOMAIN, hass)
|
||||
@ -102,7 +136,9 @@ async def async_setup(hass: HomeAssistant, config: ConfigType) -> bool:
|
||||
)
|
||||
|
||||
storage_collection = InputSelectStorageCollection(
|
||||
Store(hass, STORAGE_VERSION, STORAGE_KEY),
|
||||
InputSelectStore(
|
||||
hass, STORAGE_VERSION, STORAGE_KEY, minor_version=STORAGE_VERSION_MINOR
|
||||
),
|
||||
logging.getLogger(f"{__name__}.storage_collection"),
|
||||
id_manager,
|
||||
)
|
||||
@ -301,6 +337,10 @@ class InputSelect(SelectEntity, RestoreEntity):
|
||||
|
||||
async def async_set_options(self, options: list[str]) -> None:
|
||||
"""Set options."""
|
||||
unique_options = list(dict.fromkeys(options))
|
||||
if len(unique_options) != len(options):
|
||||
raise HomeAssistantError(f"Duplicated options: {options}")
|
||||
|
||||
self._attr_options = options
|
||||
|
||||
if self.current_option not in self.options:
|
||||
|
@ -15,6 +15,8 @@ from homeassistant.components.input_select import (
|
||||
SERVICE_SELECT_OPTION,
|
||||
SERVICE_SELECT_PREVIOUS,
|
||||
SERVICE_SET_OPTIONS,
|
||||
STORAGE_VERSION,
|
||||
STORAGE_VERSION_MINOR,
|
||||
)
|
||||
from homeassistant.const import (
|
||||
ATTR_EDITABLE,
|
||||
@ -25,7 +27,7 @@ from homeassistant.const import (
|
||||
SERVICE_RELOAD,
|
||||
)
|
||||
from homeassistant.core import Context, State
|
||||
from homeassistant.exceptions import Unauthorized
|
||||
from homeassistant.exceptions import HomeAssistantError, Unauthorized
|
||||
from homeassistant.helpers import entity_registry as er
|
||||
from homeassistant.setup import async_setup_component
|
||||
|
||||
@ -36,11 +38,12 @@ from tests.common import mock_restore_cache
|
||||
def storage_setup(hass, hass_storage):
|
||||
"""Storage setup."""
|
||||
|
||||
async def _storage(items=None, config=None):
|
||||
async def _storage(items=None, config=None, minor_version=STORAGE_VERSION_MINOR):
|
||||
if items is None:
|
||||
hass_storage[DOMAIN] = {
|
||||
"key": DOMAIN,
|
||||
"version": 1,
|
||||
"version": STORAGE_VERSION,
|
||||
"minor_version": minor_version,
|
||||
"data": {
|
||||
"items": [
|
||||
{
|
||||
@ -55,6 +58,7 @@ def storage_setup(hass, hass_storage):
|
||||
hass_storage[DOMAIN] = {
|
||||
"key": DOMAIN,
|
||||
"version": 1,
|
||||
"minor_version": minor_version,
|
||||
"data": {"items": items},
|
||||
}
|
||||
if config is None:
|
||||
@ -320,6 +324,46 @@ async def test_set_options_service(hass):
|
||||
assert state.state == "test2"
|
||||
|
||||
|
||||
async def test_set_options_service_duplicate(hass):
|
||||
"""Test set_options service with duplicates."""
|
||||
assert await async_setup_component(
|
||||
hass,
|
||||
DOMAIN,
|
||||
{
|
||||
DOMAIN: {
|
||||
"test_1": {
|
||||
"options": ["first option", "middle option", "last option"],
|
||||
"initial": "middle option",
|
||||
}
|
||||
}
|
||||
},
|
||||
)
|
||||
entity_id = "input_select.test_1"
|
||||
|
||||
state = hass.states.get(entity_id)
|
||||
assert state.state == "middle option"
|
||||
assert state.attributes[ATTR_OPTIONS] == [
|
||||
"first option",
|
||||
"middle option",
|
||||
"last option",
|
||||
]
|
||||
|
||||
with pytest.raises(HomeAssistantError):
|
||||
await hass.services.async_call(
|
||||
DOMAIN,
|
||||
SERVICE_SET_OPTIONS,
|
||||
{ATTR_OPTIONS: ["option1", "option1"], ATTR_ENTITY_ID: entity_id},
|
||||
blocking=True,
|
||||
)
|
||||
state = hass.states.get(entity_id)
|
||||
assert state.state == "middle option"
|
||||
assert state.attributes[ATTR_OPTIONS] == [
|
||||
"first option",
|
||||
"middle option",
|
||||
"last option",
|
||||
]
|
||||
|
||||
|
||||
async def test_restore_state(hass):
|
||||
"""Ensure states are restored on startup."""
|
||||
mock_restore_cache(
|
||||
@ -488,6 +532,34 @@ async def test_load_from_storage(hass, storage_setup):
|
||||
assert state.state == "storage option 1"
|
||||
assert state.attributes.get(ATTR_FRIENDLY_NAME) == "from storage"
|
||||
assert state.attributes.get(ATTR_EDITABLE)
|
||||
assert state.attributes.get(ATTR_OPTIONS) == [
|
||||
"storage option 1",
|
||||
"storage option 2",
|
||||
]
|
||||
|
||||
|
||||
async def test_load_from_storage_duplicate(hass, storage_setup, caplog):
|
||||
"""Test set up from old storage with duplicates."""
|
||||
items = [
|
||||
{
|
||||
"id": "from_storage",
|
||||
"name": "from storage",
|
||||
"options": ["yaml update 1", "yaml update 2", "yaml update 2"],
|
||||
}
|
||||
]
|
||||
assert await storage_setup(items, minor_version=1)
|
||||
|
||||
assert (
|
||||
"Input select 'from storage' with options "
|
||||
"['yaml update 1', 'yaml update 2', 'yaml update 2'] "
|
||||
"had duplicated options, the duplicates have been removed"
|
||||
) in caplog.text
|
||||
|
||||
state = hass.states.get(f"{DOMAIN}.from_storage")
|
||||
assert state.state == "yaml update 1"
|
||||
assert state.attributes.get(ATTR_FRIENDLY_NAME) == "from storage"
|
||||
assert state.attributes.get(ATTR_EDITABLE)
|
||||
assert state.attributes.get(ATTR_OPTIONS) == ["yaml update 1", "yaml update 2"]
|
||||
|
||||
|
||||
async def test_editable_state_attribute(hass, storage_setup):
|
||||
@ -554,7 +626,7 @@ async def test_ws_delete(hass, hass_ws_client, storage_setup):
|
||||
|
||||
|
||||
async def test_update(hass, hass_ws_client, storage_setup):
|
||||
"""Test updating min/max updates the state."""
|
||||
"""Test updating options updates the state."""
|
||||
|
||||
items = [
|
||||
{
|
||||
@ -590,6 +662,7 @@ async def test_update(hass, hass_ws_client, storage_setup):
|
||||
state = hass.states.get(input_entity_id)
|
||||
assert state.attributes[ATTR_OPTIONS] == ["new option", "newer option"]
|
||||
|
||||
# Should fail because the initial state is now invalid
|
||||
await client.send_json(
|
||||
{
|
||||
"id": 7,
|
||||
@ -602,6 +675,50 @@ async def test_update(hass, hass_ws_client, storage_setup):
|
||||
assert not resp["success"]
|
||||
|
||||
|
||||
async def test_update_duplicates(hass, hass_ws_client, storage_setup, caplog):
|
||||
"""Test updating options updates the state."""
|
||||
|
||||
items = [
|
||||
{
|
||||
"id": "from_storage",
|
||||
"name": "from storage",
|
||||
"options": ["yaml update 1", "yaml update 2"],
|
||||
}
|
||||
]
|
||||
assert await storage_setup(items)
|
||||
|
||||
input_id = "from_storage"
|
||||
input_entity_id = f"{DOMAIN}.{input_id}"
|
||||
ent_reg = er.async_get(hass)
|
||||
|
||||
state = hass.states.get(input_entity_id)
|
||||
assert state.attributes[ATTR_OPTIONS] == ["yaml update 1", "yaml update 2"]
|
||||
assert ent_reg.async_get_entity_id(DOMAIN, DOMAIN, input_id) is not None
|
||||
|
||||
client = await hass_ws_client(hass)
|
||||
|
||||
await client.send_json(
|
||||
{
|
||||
"id": 6,
|
||||
"type": f"{DOMAIN}/update",
|
||||
f"{DOMAIN}_id": f"{input_id}",
|
||||
"options": ["new option", "newer option", "newer option"],
|
||||
CONF_INITIAL: "newer option",
|
||||
}
|
||||
)
|
||||
resp = await client.receive_json()
|
||||
assert resp["success"]
|
||||
|
||||
assert (
|
||||
"Input select 'from storage' with options "
|
||||
"['new option', 'newer option', 'newer option'] "
|
||||
"had duplicated options, the duplicates have been removed"
|
||||
) in caplog.text
|
||||
|
||||
state = hass.states.get(input_entity_id)
|
||||
assert state.attributes[ATTR_OPTIONS] == ["new option", "newer option"]
|
||||
|
||||
|
||||
async def test_ws_create(hass, hass_ws_client, storage_setup):
|
||||
"""Test create WS."""
|
||||
assert await storage_setup(items=[])
|
||||
@ -630,6 +747,44 @@ async def test_ws_create(hass, hass_ws_client, storage_setup):
|
||||
|
||||
state = hass.states.get(input_entity_id)
|
||||
assert state.state == "even newer option"
|
||||
assert state.attributes[ATTR_OPTIONS] == ["new option", "even newer option"]
|
||||
|
||||
|
||||
async def test_ws_create_duplicates(hass, hass_ws_client, storage_setup, caplog):
|
||||
"""Test create WS with duplicates."""
|
||||
assert await storage_setup(items=[])
|
||||
|
||||
input_id = "new_input"
|
||||
input_entity_id = f"{DOMAIN}.{input_id}"
|
||||
ent_reg = er.async_get(hass)
|
||||
|
||||
state = hass.states.get(input_entity_id)
|
||||
assert state is None
|
||||
assert ent_reg.async_get_entity_id(DOMAIN, DOMAIN, input_id) is None
|
||||
|
||||
client = await hass_ws_client(hass)
|
||||
|
||||
await client.send_json(
|
||||
{
|
||||
"id": 6,
|
||||
"type": f"{DOMAIN}/create",
|
||||
"name": "New Input",
|
||||
"options": ["new option", "even newer option", "even newer option"],
|
||||
"initial": "even newer option",
|
||||
}
|
||||
)
|
||||
resp = await client.receive_json()
|
||||
assert resp["success"]
|
||||
|
||||
assert (
|
||||
"Input select 'New Input' with options "
|
||||
"['new option', 'even newer option', 'even newer option'] "
|
||||
"had duplicated options, the duplicates have been removed"
|
||||
) in caplog.text
|
||||
|
||||
state = hass.states.get(input_entity_id)
|
||||
assert state.state == "even newer option"
|
||||
assert state.attributes[ATTR_OPTIONS] == ["new option", "even newer option"]
|
||||
|
||||
|
||||
async def test_setup_no_config(hass, hass_admin_user):
|
||||
|
Loading…
x
Reference in New Issue
Block a user