mirror of
https://github.com/home-assistant/core.git
synced 2025-07-18 18:57:06 +00:00
Add voluptuous config validation to scenes (#6830)
* Add platform schema to scene component and homeassistant platform. * Clean up code and add constants. * Add unit test and clean up tests.
This commit is contained in:
parent
d1b519a418
commit
7c614a6738
@ -6,28 +6,55 @@ https://home-assistant.io/components/scene/
|
|||||||
"""
|
"""
|
||||||
import asyncio
|
import asyncio
|
||||||
import logging
|
import logging
|
||||||
from collections import namedtuple
|
|
||||||
|
|
||||||
import voluptuous as vol
|
import voluptuous as vol
|
||||||
|
|
||||||
from homeassistant.const import (
|
from homeassistant.const import (
|
||||||
ATTR_ENTITY_ID, SERVICE_TURN_ON, CONF_PLATFORM)
|
ATTR_ENTITY_ID, CONF_PLATFORM, SERVICE_TURN_ON)
|
||||||
from homeassistant.helpers import extract_domain_configs
|
|
||||||
import homeassistant.helpers.config_validation as cv
|
import homeassistant.helpers.config_validation as cv
|
||||||
from homeassistant.helpers.entity import Entity
|
from homeassistant.helpers.entity import Entity
|
||||||
from homeassistant.helpers.entity_component import EntityComponent
|
from homeassistant.helpers.entity_component import EntityComponent
|
||||||
|
from homeassistant.helpers.state import HASS_DOMAIN
|
||||||
|
from homeassistant.loader import get_platform
|
||||||
|
|
||||||
DOMAIN = 'scene'
|
DOMAIN = 'scene'
|
||||||
STATE = 'scening'
|
STATE = 'scening'
|
||||||
|
STATES = 'states'
|
||||||
|
|
||||||
CONF_ENTITIES = "entities"
|
|
||||||
|
def _hass_domain_validator(config):
|
||||||
|
"""Validate platform in config for homeassistant domain."""
|
||||||
|
if CONF_PLATFORM not in config:
|
||||||
|
config = {
|
||||||
|
CONF_PLATFORM: HASS_DOMAIN, STATES: config}
|
||||||
|
|
||||||
|
return config
|
||||||
|
|
||||||
|
|
||||||
|
def _platform_validator(config):
|
||||||
|
"""Validate it is a valid platform."""
|
||||||
|
p_name = config[CONF_PLATFORM]
|
||||||
|
platform = get_platform(DOMAIN, p_name)
|
||||||
|
|
||||||
|
if not hasattr(platform, 'PLATFORM_SCHEMA'):
|
||||||
|
return config
|
||||||
|
|
||||||
|
return getattr(platform, 'PLATFORM_SCHEMA')(config)
|
||||||
|
|
||||||
|
|
||||||
|
PLATFORM_SCHEMA = vol.Schema(
|
||||||
|
vol.All(
|
||||||
|
_hass_domain_validator,
|
||||||
|
vol.Schema({
|
||||||
|
vol.Required(CONF_PLATFORM): cv.platform_validator(DOMAIN)
|
||||||
|
}, extra=vol.ALLOW_EXTRA),
|
||||||
|
_platform_validator
|
||||||
|
), extra=vol.ALLOW_EXTRA)
|
||||||
|
|
||||||
SCENE_SERVICE_SCHEMA = vol.Schema({
|
SCENE_SERVICE_SCHEMA = vol.Schema({
|
||||||
vol.Optional(ATTR_ENTITY_ID): cv.entity_ids,
|
vol.Optional(ATTR_ENTITY_ID): cv.entity_ids,
|
||||||
})
|
})
|
||||||
|
|
||||||
SceneConfig = namedtuple('SceneConfig', ['name', 'states'])
|
|
||||||
|
|
||||||
|
|
||||||
def activate(hass, entity_id=None):
|
def activate(hass, entity_id=None):
|
||||||
"""Activate a scene."""
|
"""Activate a scene."""
|
||||||
@ -43,21 +70,6 @@ def activate(hass, entity_id=None):
|
|||||||
def async_setup(hass, config):
|
def async_setup(hass, config):
|
||||||
"""Setup scenes."""
|
"""Setup scenes."""
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
# You are not allowed to mutate the original config so make a copy
|
|
||||||
config = dict(config)
|
|
||||||
|
|
||||||
for config_key in extract_domain_configs(config, DOMAIN):
|
|
||||||
platform_config = config[config_key]
|
|
||||||
if not isinstance(platform_config, list):
|
|
||||||
platform_config = [platform_config]
|
|
||||||
|
|
||||||
if not any(CONF_PLATFORM in entry for entry in platform_config):
|
|
||||||
platform_config = [{'platform': 'homeassistant', 'states': entry}
|
|
||||||
for entry in platform_config]
|
|
||||||
|
|
||||||
config[config_key] = platform_config
|
|
||||||
|
|
||||||
component = EntityComponent(logger, DOMAIN, hass)
|
component = EntityComponent(logger, DOMAIN, hass)
|
||||||
|
|
||||||
yield from component.async_setup(config)
|
yield from component.async_setup(config)
|
||||||
|
@ -7,26 +7,38 @@ https://home-assistant.io/components/scene/
|
|||||||
import asyncio
|
import asyncio
|
||||||
from collections import namedtuple
|
from collections import namedtuple
|
||||||
|
|
||||||
from homeassistant.components.scene import Scene
|
import voluptuous as vol
|
||||||
|
|
||||||
|
import homeassistant.helpers.config_validation as cv
|
||||||
|
from homeassistant.components.scene import Scene, STATES
|
||||||
from homeassistant.const import (
|
from homeassistant.const import (
|
||||||
ATTR_ENTITY_ID, STATE_OFF, STATE_ON)
|
ATTR_ENTITY_ID, ATTR_STATE, CONF_ENTITIES, CONF_NAME, CONF_PLATFORM,
|
||||||
|
STATE_OFF, STATE_ON)
|
||||||
from homeassistant.core import State
|
from homeassistant.core import State
|
||||||
from homeassistant.helpers.state import async_reproduce_state
|
from homeassistant.helpers.state import async_reproduce_state, HASS_DOMAIN
|
||||||
|
|
||||||
STATE = 'scening'
|
PLATFORM_SCHEMA = vol.Schema({
|
||||||
|
vol.Required(CONF_PLATFORM): HASS_DOMAIN,
|
||||||
|
vol.Required(STATES): vol.All(
|
||||||
|
cv.ensure_list,
|
||||||
|
[
|
||||||
|
{
|
||||||
|
vol.Required(CONF_NAME): cv.string,
|
||||||
|
vol.Required(CONF_ENTITIES): {
|
||||||
|
cv.entity_id: vol.Any(str, bool, dict)
|
||||||
|
},
|
||||||
|
}
|
||||||
|
]
|
||||||
|
),
|
||||||
|
}, extra=vol.ALLOW_EXTRA)
|
||||||
|
|
||||||
CONF_ENTITIES = "entities"
|
SCENECONFIG = namedtuple('SceneConfig', [CONF_NAME, STATES])
|
||||||
|
|
||||||
SceneConfig = namedtuple('SceneConfig', ['name', 'states'])
|
|
||||||
|
|
||||||
|
|
||||||
@asyncio.coroutine
|
@asyncio.coroutine
|
||||||
def async_setup_platform(hass, config, async_add_devices, discovery_info=None):
|
def async_setup_platform(hass, config, async_add_devices, discovery_info=None):
|
||||||
"""Setup home assistant scene entries."""
|
"""Setup home assistant scene entries."""
|
||||||
scene_config = config.get("states")
|
scene_config = config.get(STATES)
|
||||||
|
|
||||||
if not isinstance(scene_config, list):
|
|
||||||
scene_config = [scene_config]
|
|
||||||
|
|
||||||
async_add_devices(HomeAssistantScene(
|
async_add_devices(HomeAssistantScene(
|
||||||
hass, _process_config(scene)) for scene in scene_config)
|
hass, _process_config(scene)) for scene in scene_config)
|
||||||
@ -38,7 +50,7 @@ def _process_config(scene_config):
|
|||||||
|
|
||||||
Async friendly.
|
Async friendly.
|
||||||
"""
|
"""
|
||||||
name = scene_config.get('name')
|
name = scene_config.get(CONF_NAME)
|
||||||
|
|
||||||
states = {}
|
states = {}
|
||||||
c_entities = dict(scene_config.get(CONF_ENTITIES, {}))
|
c_entities = dict(scene_config.get(CONF_ENTITIES, {}))
|
||||||
@ -46,7 +58,7 @@ def _process_config(scene_config):
|
|||||||
for entity_id in c_entities:
|
for entity_id in c_entities:
|
||||||
if isinstance(c_entities[entity_id], dict):
|
if isinstance(c_entities[entity_id], dict):
|
||||||
entity_attrs = c_entities[entity_id].copy()
|
entity_attrs = c_entities[entity_id].copy()
|
||||||
state = entity_attrs.pop('state', None)
|
state = entity_attrs.pop(ATTR_STATE, None)
|
||||||
attributes = entity_attrs
|
attributes = entity_attrs
|
||||||
else:
|
else:
|
||||||
state = c_entities[entity_id]
|
state = c_entities[entity_id]
|
||||||
@ -61,7 +73,7 @@ def _process_config(scene_config):
|
|||||||
|
|
||||||
states[entity_id.lower()] = State(entity_id, state, attributes)
|
states[entity_id.lower()] = State(entity_id, state, attributes)
|
||||||
|
|
||||||
return SceneConfig(name, states)
|
return SCENECONFIG(name, states)
|
||||||
|
|
||||||
|
|
||||||
class HomeAssistantScene(Scene):
|
class HomeAssistantScene(Scene):
|
||||||
|
@ -1,9 +1,11 @@
|
|||||||
"""The tests for the Scene component."""
|
"""The tests for the Scene component."""
|
||||||
|
import io
|
||||||
import unittest
|
import unittest
|
||||||
|
|
||||||
from homeassistant.setup import setup_component
|
from homeassistant.setup import setup_component
|
||||||
from homeassistant import loader
|
from homeassistant import loader
|
||||||
from homeassistant.components import light, scene
|
from homeassistant.components import light, scene
|
||||||
|
from homeassistant.util import yaml
|
||||||
|
|
||||||
from tests.common import get_test_home_assistant
|
from tests.common import get_test_home_assistant
|
||||||
|
|
||||||
@ -14,6 +16,22 @@ class TestScene(unittest.TestCase):
|
|||||||
def setUp(self): # pylint: disable=invalid-name
|
def setUp(self): # pylint: disable=invalid-name
|
||||||
"""Setup things to be run when tests are started."""
|
"""Setup things to be run when tests are started."""
|
||||||
self.hass = get_test_home_assistant()
|
self.hass = get_test_home_assistant()
|
||||||
|
test_light = loader.get_component('light.test')
|
||||||
|
test_light.init()
|
||||||
|
|
||||||
|
self.assertTrue(setup_component(self.hass, light.DOMAIN, {
|
||||||
|
light.DOMAIN: {'platform': 'test'}
|
||||||
|
}))
|
||||||
|
|
||||||
|
self.light_1, self.light_2 = test_light.DEVICES[0:2]
|
||||||
|
|
||||||
|
light.turn_off(
|
||||||
|
self.hass, [self.light_1.entity_id, self.light_2.entity_id])
|
||||||
|
|
||||||
|
self.hass.block_till_done()
|
||||||
|
|
||||||
|
self.assertFalse(self.light_1.is_on)
|
||||||
|
self.assertFalse(self.light_2.is_on)
|
||||||
|
|
||||||
def tearDown(self): # pylint: disable=invalid-name
|
def tearDown(self): # pylint: disable=invalid-name
|
||||||
"""Stop everything that was started."""
|
"""Stop everything that was started."""
|
||||||
@ -36,19 +54,6 @@ class TestScene(unittest.TestCase):
|
|||||||
reference to the original dictionary, instead of creating a copy, so
|
reference to the original dictionary, instead of creating a copy, so
|
||||||
care needs to be taken to not modify the original.
|
care needs to be taken to not modify the original.
|
||||||
"""
|
"""
|
||||||
test_light = loader.get_component('light.test')
|
|
||||||
test_light.init()
|
|
||||||
|
|
||||||
self.assertTrue(setup_component(self.hass, light.DOMAIN, {
|
|
||||||
light.DOMAIN: {'platform': 'test'}
|
|
||||||
}))
|
|
||||||
|
|
||||||
light_1, light_2 = test_light.DEVICES[0:2]
|
|
||||||
|
|
||||||
light.turn_off(self.hass, [light_1.entity_id, light_2.entity_id])
|
|
||||||
|
|
||||||
self.hass.block_till_done()
|
|
||||||
|
|
||||||
entity_state = {
|
entity_state = {
|
||||||
'state': 'on',
|
'state': 'on',
|
||||||
'brightness': 100,
|
'brightness': 100,
|
||||||
@ -57,8 +62,8 @@ class TestScene(unittest.TestCase):
|
|||||||
'scene': [{
|
'scene': [{
|
||||||
'name': 'test',
|
'name': 'test',
|
||||||
'entities': {
|
'entities': {
|
||||||
light_1.entity_id: entity_state,
|
self.light_1.entity_id: entity_state,
|
||||||
light_2.entity_id: entity_state,
|
self.light_2.entity_id: entity_state,
|
||||||
}
|
}
|
||||||
}]
|
}]
|
||||||
}))
|
}))
|
||||||
@ -66,34 +71,45 @@ class TestScene(unittest.TestCase):
|
|||||||
scene.activate(self.hass, 'scene.test')
|
scene.activate(self.hass, 'scene.test')
|
||||||
self.hass.block_till_done()
|
self.hass.block_till_done()
|
||||||
|
|
||||||
self.assertTrue(light_1.is_on)
|
self.assertTrue(self.light_1.is_on)
|
||||||
self.assertTrue(light_2.is_on)
|
self.assertTrue(self.light_2.is_on)
|
||||||
self.assertEqual(100,
|
self.assertEqual(
|
||||||
light_1.last_call('turn_on')[1].get('brightness'))
|
100, self.light_1.last_call('turn_on')[1].get('brightness'))
|
||||||
self.assertEqual(100,
|
self.assertEqual(
|
||||||
light_2.last_call('turn_on')[1].get('brightness'))
|
100, self.light_2.last_call('turn_on')[1].get('brightness'))
|
||||||
|
|
||||||
|
def test_config_yaml_bool(self):
|
||||||
|
"""Test parsing of booleans in yaml config."""
|
||||||
|
config = (
|
||||||
|
'scene:\n'
|
||||||
|
' - name: test\n'
|
||||||
|
' entities:\n'
|
||||||
|
' {0}: on\n'
|
||||||
|
' {1}:\n'
|
||||||
|
' state: on\n'
|
||||||
|
' brightness: 100\n').format(
|
||||||
|
self.light_1.entity_id, self.light_2.entity_id)
|
||||||
|
|
||||||
|
with io.StringIO(config) as file:
|
||||||
|
doc = yaml.yaml.safe_load(file)
|
||||||
|
|
||||||
|
self.assertTrue(setup_component(self.hass, scene.DOMAIN, doc))
|
||||||
|
scene.activate(self.hass, 'scene.test')
|
||||||
|
self.hass.block_till_done()
|
||||||
|
|
||||||
|
self.assertTrue(self.light_1.is_on)
|
||||||
|
self.assertTrue(self.light_2.is_on)
|
||||||
|
self.assertEqual(
|
||||||
|
100, self.light_2.last_call('turn_on')[1].get('brightness'))
|
||||||
|
|
||||||
def test_activate_scene(self):
|
def test_activate_scene(self):
|
||||||
"""Test active scene."""
|
"""Test active scene."""
|
||||||
test_light = loader.get_component('light.test')
|
|
||||||
test_light.init()
|
|
||||||
|
|
||||||
self.assertTrue(setup_component(self.hass, light.DOMAIN, {
|
|
||||||
light.DOMAIN: {'platform': 'test'}
|
|
||||||
}))
|
|
||||||
|
|
||||||
light_1, light_2 = test_light.DEVICES[0:2]
|
|
||||||
|
|
||||||
light.turn_off(self.hass, [light_1.entity_id, light_2.entity_id])
|
|
||||||
|
|
||||||
self.hass.block_till_done()
|
|
||||||
|
|
||||||
self.assertTrue(setup_component(self.hass, scene.DOMAIN, {
|
self.assertTrue(setup_component(self.hass, scene.DOMAIN, {
|
||||||
'scene': [{
|
'scene': [{
|
||||||
'name': 'test',
|
'name': 'test',
|
||||||
'entities': {
|
'entities': {
|
||||||
light_1.entity_id: 'on',
|
self.light_1.entity_id: 'on',
|
||||||
light_2.entity_id: {
|
self.light_2.entity_id: {
|
||||||
'state': 'on',
|
'state': 'on',
|
||||||
'brightness': 100,
|
'brightness': 100,
|
||||||
}
|
}
|
||||||
@ -104,7 +120,7 @@ class TestScene(unittest.TestCase):
|
|||||||
scene.activate(self.hass, 'scene.test')
|
scene.activate(self.hass, 'scene.test')
|
||||||
self.hass.block_till_done()
|
self.hass.block_till_done()
|
||||||
|
|
||||||
self.assertTrue(light_1.is_on)
|
self.assertTrue(self.light_1.is_on)
|
||||||
self.assertTrue(light_2.is_on)
|
self.assertTrue(self.light_2.is_on)
|
||||||
self.assertEqual(100,
|
self.assertEqual(
|
||||||
light_2.last_call('turn_on')[1].get('brightness'))
|
100, self.light_2.last_call('turn_on')[1].get('brightness'))
|
||||||
|
Loading…
x
Reference in New Issue
Block a user