mirror of
https://github.com/home-assistant/core.git
synced 2025-07-17 18:27:09 +00:00
Add counter component (#9146)
This commit is contained in:
parent
38071501b4
commit
0687a457b1
220
homeassistant/components/counter.py
Normal file
220
homeassistant/components/counter.py
Normal file
@ -0,0 +1,220 @@
|
|||||||
|
"""
|
||||||
|
Component to count within automations.
|
||||||
|
|
||||||
|
For more details about this component, please refer to the documentation
|
||||||
|
at https://home-assistant.io/components/counter/
|
||||||
|
"""
|
||||||
|
import asyncio
|
||||||
|
import logging
|
||||||
|
import os
|
||||||
|
|
||||||
|
import voluptuous as vol
|
||||||
|
|
||||||
|
import homeassistant.helpers.config_validation as cv
|
||||||
|
from homeassistant.config import load_yaml_config_file
|
||||||
|
from homeassistant.const import (ATTR_ENTITY_ID, CONF_ICON, CONF_NAME)
|
||||||
|
from homeassistant.core import callback
|
||||||
|
from homeassistant.helpers.entity import Entity
|
||||||
|
from homeassistant.helpers.entity_component import EntityComponent
|
||||||
|
from homeassistant.helpers.restore_state import async_get_last_state
|
||||||
|
from homeassistant.loader import bind_hass
|
||||||
|
|
||||||
|
_LOGGER = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
ATTR_INITIAL = 'initial'
|
||||||
|
ATTR_STEP = 'step'
|
||||||
|
|
||||||
|
CONF_INITIAL = 'initial'
|
||||||
|
CONF_STEP = 'step'
|
||||||
|
|
||||||
|
DEFAULT_INITIAL = 0
|
||||||
|
DEFAULT_STEP = 1
|
||||||
|
DOMAIN = 'counter'
|
||||||
|
|
||||||
|
ENTITY_ID_FORMAT = DOMAIN + '.{}'
|
||||||
|
|
||||||
|
SERVICE_DECREMENT = 'decrement'
|
||||||
|
SERVICE_INCREMENT = 'increment'
|
||||||
|
SERVICE_RESET = 'reset'
|
||||||
|
|
||||||
|
SERVICE_SCHEMA = vol.Schema({
|
||||||
|
vol.Optional(ATTR_ENTITY_ID): cv.entity_ids,
|
||||||
|
})
|
||||||
|
|
||||||
|
CONFIG_SCHEMA = vol.Schema({
|
||||||
|
DOMAIN: vol.Schema({
|
||||||
|
cv.slug: vol.Any({
|
||||||
|
vol.Optional(CONF_ICON): cv.icon,
|
||||||
|
vol.Optional(CONF_INITIAL, default=DEFAULT_INITIAL):
|
||||||
|
cv.positive_int,
|
||||||
|
vol.Optional(CONF_NAME): cv.string,
|
||||||
|
vol.Optional(CONF_STEP, default=DEFAULT_STEP): cv.positive_int,
|
||||||
|
}, None)
|
||||||
|
})
|
||||||
|
}, extra=vol.ALLOW_EXTRA)
|
||||||
|
|
||||||
|
|
||||||
|
@bind_hass
|
||||||
|
def increment(hass, entity_id):
|
||||||
|
"""Increment a counter."""
|
||||||
|
hass.add_job(async_increment, hass, entity_id)
|
||||||
|
|
||||||
|
|
||||||
|
@callback
|
||||||
|
@bind_hass
|
||||||
|
def async_increment(hass, entity_id):
|
||||||
|
"""Increment a counter."""
|
||||||
|
hass.async_add_job(hass.services.async_call(
|
||||||
|
DOMAIN, SERVICE_INCREMENT, {ATTR_ENTITY_ID: entity_id}))
|
||||||
|
|
||||||
|
|
||||||
|
@bind_hass
|
||||||
|
def decrement(hass, entity_id):
|
||||||
|
"""Decrement a counter."""
|
||||||
|
hass.add_job(async_decrement, hass, entity_id)
|
||||||
|
|
||||||
|
|
||||||
|
@callback
|
||||||
|
@bind_hass
|
||||||
|
def async_decrement(hass, entity_id):
|
||||||
|
"""Decrement a counter."""
|
||||||
|
hass.async_add_job(hass.services.async_call(
|
||||||
|
DOMAIN, SERVICE_DECREMENT, {ATTR_ENTITY_ID: entity_id}))
|
||||||
|
|
||||||
|
|
||||||
|
@bind_hass
|
||||||
|
def reset(hass, entity_id):
|
||||||
|
"""Reset a counter."""
|
||||||
|
hass.add_job(async_reset, hass, entity_id)
|
||||||
|
|
||||||
|
|
||||||
|
@callback
|
||||||
|
@bind_hass
|
||||||
|
def async_reset(hass, entity_id):
|
||||||
|
"""Reset a counter."""
|
||||||
|
hass.async_add_job(hass.services.async_call(
|
||||||
|
DOMAIN, SERVICE_RESET, {ATTR_ENTITY_ID: entity_id}))
|
||||||
|
|
||||||
|
|
||||||
|
@asyncio.coroutine
|
||||||
|
def async_setup(hass, config):
|
||||||
|
"""Set up a counter."""
|
||||||
|
component = EntityComponent(_LOGGER, DOMAIN, hass)
|
||||||
|
|
||||||
|
entities = []
|
||||||
|
|
||||||
|
for object_id, cfg in config[DOMAIN].items():
|
||||||
|
if not cfg:
|
||||||
|
cfg = {}
|
||||||
|
|
||||||
|
name = cfg.get(CONF_NAME)
|
||||||
|
initial = cfg.get(CONF_INITIAL)
|
||||||
|
step = cfg.get(CONF_STEP)
|
||||||
|
icon = cfg.get(CONF_ICON)
|
||||||
|
|
||||||
|
entities.append(Counter(object_id, name, initial, step, icon))
|
||||||
|
|
||||||
|
if not entities:
|
||||||
|
return False
|
||||||
|
|
||||||
|
@asyncio.coroutine
|
||||||
|
def async_handler_service(service):
|
||||||
|
"""Handle a call to the counter services."""
|
||||||
|
target_counters = component.async_extract_from_service(service)
|
||||||
|
|
||||||
|
if service.service == SERVICE_INCREMENT:
|
||||||
|
attr = 'async_increment'
|
||||||
|
elif service.service == SERVICE_DECREMENT:
|
||||||
|
attr = 'async_decrement'
|
||||||
|
elif service.service == SERVICE_RESET:
|
||||||
|
attr = 'async_reset'
|
||||||
|
|
||||||
|
tasks = [getattr(counter, attr)() for counter in target_counters]
|
||||||
|
if tasks:
|
||||||
|
yield from asyncio.wait(tasks, loop=hass.loop)
|
||||||
|
|
||||||
|
descriptions = yield from hass.async_add_job(
|
||||||
|
load_yaml_config_file, os.path.join(
|
||||||
|
os.path.dirname(__file__), 'services.yaml')
|
||||||
|
)
|
||||||
|
|
||||||
|
hass.services.async_register(
|
||||||
|
DOMAIN, SERVICE_INCREMENT, async_handler_service,
|
||||||
|
descriptions[DOMAIN][SERVICE_INCREMENT], SERVICE_SCHEMA)
|
||||||
|
hass.services.async_register(
|
||||||
|
DOMAIN, SERVICE_DECREMENT, async_handler_service,
|
||||||
|
descriptions[DOMAIN][SERVICE_DECREMENT], SERVICE_SCHEMA)
|
||||||
|
hass.services.async_register(
|
||||||
|
DOMAIN, SERVICE_RESET, async_handler_service,
|
||||||
|
descriptions[DOMAIN][SERVICE_RESET], SERVICE_SCHEMA)
|
||||||
|
|
||||||
|
yield from component.async_add_entities(entities)
|
||||||
|
return True
|
||||||
|
|
||||||
|
|
||||||
|
class Counter(Entity):
|
||||||
|
"""Representation of a counter."""
|
||||||
|
|
||||||
|
def __init__(self, object_id, name, initial, step, icon):
|
||||||
|
"""Initialize a counter."""
|
||||||
|
self.entity_id = ENTITY_ID_FORMAT.format(object_id)
|
||||||
|
self._name = name
|
||||||
|
self._step = step
|
||||||
|
self._state = self._initial = initial
|
||||||
|
self._icon = icon
|
||||||
|
|
||||||
|
@property
|
||||||
|
def should_poll(self):
|
||||||
|
"""If entity should be polled."""
|
||||||
|
return False
|
||||||
|
|
||||||
|
@property
|
||||||
|
def name(self):
|
||||||
|
"""Return name of the counter."""
|
||||||
|
return self._name
|
||||||
|
|
||||||
|
@property
|
||||||
|
def icon(self):
|
||||||
|
"""Return the icon to be used for this entity."""
|
||||||
|
return self._icon
|
||||||
|
|
||||||
|
@property
|
||||||
|
def state(self):
|
||||||
|
"""Return the current value of the counter."""
|
||||||
|
return self._state
|
||||||
|
|
||||||
|
@property
|
||||||
|
def state_attributes(self):
|
||||||
|
"""Return the state attributes."""
|
||||||
|
return {
|
||||||
|
ATTR_INITIAL: self._initial,
|
||||||
|
ATTR_STEP: self._step,
|
||||||
|
}
|
||||||
|
|
||||||
|
@asyncio.coroutine
|
||||||
|
def async_added_to_hass(self):
|
||||||
|
"""Call when entity about to be added to Home Assistant."""
|
||||||
|
# If not None, we got an initial value.
|
||||||
|
if self._state is not None:
|
||||||
|
return
|
||||||
|
|
||||||
|
state = yield from async_get_last_state(self.hass, self.entity_id)
|
||||||
|
self._state = state and state.state == state
|
||||||
|
|
||||||
|
@asyncio.coroutine
|
||||||
|
def async_decrement(self):
|
||||||
|
"""Decrement the counter."""
|
||||||
|
self._state -= self._step
|
||||||
|
yield from self.async_update_ha_state()
|
||||||
|
|
||||||
|
@asyncio.coroutine
|
||||||
|
def async_increment(self):
|
||||||
|
"""Increment a counter."""
|
||||||
|
self._state += self._step
|
||||||
|
yield from self.async_update_ha_state()
|
||||||
|
|
||||||
|
@asyncio.coroutine
|
||||||
|
def async_reset(self):
|
||||||
|
"""Reset a counter."""
|
||||||
|
self._state = self._initial
|
||||||
|
yield from self.async_update_ha_state()
|
@ -546,3 +546,28 @@ rflink:
|
|||||||
command:
|
command:
|
||||||
description: The command to be sent
|
description: The command to be sent
|
||||||
example: 'on'
|
example: 'on'
|
||||||
|
|
||||||
|
counter:
|
||||||
|
decrement:
|
||||||
|
description: Decrement a counter.
|
||||||
|
|
||||||
|
fields:
|
||||||
|
entity_id:
|
||||||
|
description: Entity id of the counter to decrement.
|
||||||
|
example: 'counter.count0'
|
||||||
|
|
||||||
|
increment:
|
||||||
|
description: Increment a counter.
|
||||||
|
|
||||||
|
fields:
|
||||||
|
entity_id:
|
||||||
|
description: Entity id of the counter to increment.
|
||||||
|
example: 'counter.count0'
|
||||||
|
|
||||||
|
reset:
|
||||||
|
description: Reset a counter.
|
||||||
|
|
||||||
|
fields:
|
||||||
|
entity_id:
|
||||||
|
description: Entity id of the counter to reset.
|
||||||
|
example: 'counter.count0'
|
||||||
|
204
tests/components/test_counter.py
Normal file
204
tests/components/test_counter.py
Normal file
@ -0,0 +1,204 @@
|
|||||||
|
"""The tests for the counter component."""
|
||||||
|
# pylint: disable=protected-access
|
||||||
|
import asyncio
|
||||||
|
import unittest
|
||||||
|
import logging
|
||||||
|
|
||||||
|
from homeassistant.core import CoreState, State
|
||||||
|
from homeassistant.setup import setup_component, async_setup_component
|
||||||
|
from homeassistant.components.counter import (
|
||||||
|
DOMAIN, decrement, increment, reset, CONF_INITIAL, CONF_STEP, CONF_NAME,
|
||||||
|
CONF_ICON)
|
||||||
|
from homeassistant.const import (ATTR_ICON, ATTR_FRIENDLY_NAME)
|
||||||
|
|
||||||
|
from tests.common import (get_test_home_assistant, mock_restore_cache)
|
||||||
|
|
||||||
|
_LOGGER = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
class TestCounter(unittest.TestCase):
|
||||||
|
"""Test the counter component."""
|
||||||
|
|
||||||
|
# pylint: disable=invalid-name
|
||||||
|
def setUp(self):
|
||||||
|
"""Set up things to be run when tests are started."""
|
||||||
|
self.hass = get_test_home_assistant()
|
||||||
|
|
||||||
|
# pylint: disable=invalid-name
|
||||||
|
def tearDown(self):
|
||||||
|
"""Stop everything that was started."""
|
||||||
|
self.hass.stop()
|
||||||
|
|
||||||
|
def test_config(self):
|
||||||
|
"""Test config."""
|
||||||
|
invalid_configs = [
|
||||||
|
None,
|
||||||
|
1,
|
||||||
|
{},
|
||||||
|
{'name with space': None},
|
||||||
|
]
|
||||||
|
|
||||||
|
for cfg in invalid_configs:
|
||||||
|
self.assertFalse(
|
||||||
|
setup_component(self.hass, DOMAIN, {DOMAIN: cfg}))
|
||||||
|
|
||||||
|
def test_methods(self):
|
||||||
|
"""Test increment, decrement, and reset methods."""
|
||||||
|
config = {
|
||||||
|
DOMAIN: {
|
||||||
|
'test_1': {},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
assert setup_component(self.hass, 'counter', config)
|
||||||
|
|
||||||
|
entity_id = 'counter.test_1'
|
||||||
|
|
||||||
|
state = self.hass.states.get(entity_id)
|
||||||
|
self.assertEqual(0, int(state.state))
|
||||||
|
|
||||||
|
increment(self.hass, entity_id)
|
||||||
|
self.hass.block_till_done()
|
||||||
|
|
||||||
|
state = self.hass.states.get(entity_id)
|
||||||
|
self.assertEqual(1, int(state.state))
|
||||||
|
|
||||||
|
increment(self.hass, entity_id)
|
||||||
|
self.hass.block_till_done()
|
||||||
|
|
||||||
|
state = self.hass.states.get(entity_id)
|
||||||
|
self.assertEqual(2, int(state.state))
|
||||||
|
|
||||||
|
decrement(self.hass, entity_id)
|
||||||
|
self.hass.block_till_done()
|
||||||
|
|
||||||
|
state = self.hass.states.get(entity_id)
|
||||||
|
self.assertEqual(1, int(state.state))
|
||||||
|
|
||||||
|
reset(self.hass, entity_id)
|
||||||
|
self.hass.block_till_done()
|
||||||
|
|
||||||
|
state = self.hass.states.get(entity_id)
|
||||||
|
self.assertEqual(0, int(state.state))
|
||||||
|
|
||||||
|
def test_methods_with_config(self):
|
||||||
|
"""Test increment, decrement, and reset methods with configuration."""
|
||||||
|
config = {
|
||||||
|
DOMAIN: {
|
||||||
|
'test': {
|
||||||
|
CONF_NAME: 'Hello World',
|
||||||
|
CONF_INITIAL: 10,
|
||||||
|
CONF_STEP: 5,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
assert setup_component(self.hass, 'counter', config)
|
||||||
|
|
||||||
|
entity_id = 'counter.test'
|
||||||
|
|
||||||
|
state = self.hass.states.get(entity_id)
|
||||||
|
self.assertEqual(10, int(state.state))
|
||||||
|
|
||||||
|
increment(self.hass, entity_id)
|
||||||
|
self.hass.block_till_done()
|
||||||
|
|
||||||
|
state = self.hass.states.get(entity_id)
|
||||||
|
self.assertEqual(15, int(state.state))
|
||||||
|
|
||||||
|
increment(self.hass, entity_id)
|
||||||
|
self.hass.block_till_done()
|
||||||
|
|
||||||
|
state = self.hass.states.get(entity_id)
|
||||||
|
self.assertEqual(20, int(state.state))
|
||||||
|
|
||||||
|
decrement(self.hass, entity_id)
|
||||||
|
self.hass.block_till_done()
|
||||||
|
|
||||||
|
state = self.hass.states.get(entity_id)
|
||||||
|
self.assertEqual(15, int(state.state))
|
||||||
|
|
||||||
|
def test_config_options(self):
|
||||||
|
"""Test configuration options."""
|
||||||
|
count_start = len(self.hass.states.entity_ids())
|
||||||
|
|
||||||
|
_LOGGER.debug('ENTITIES @ start: %s', self.hass.states.entity_ids())
|
||||||
|
|
||||||
|
config = {
|
||||||
|
DOMAIN: {
|
||||||
|
'test_1': {},
|
||||||
|
'test_2': {
|
||||||
|
CONF_NAME: 'Hello World',
|
||||||
|
CONF_ICON: 'mdi:work',
|
||||||
|
CONF_INITIAL: 10,
|
||||||
|
CONF_STEP: 5,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
assert setup_component(self.hass, 'counter', config)
|
||||||
|
self.hass.block_till_done()
|
||||||
|
|
||||||
|
_LOGGER.debug('ENTITIES: %s', self.hass.states.entity_ids())
|
||||||
|
|
||||||
|
self.assertEqual(count_start + 2, len(self.hass.states.entity_ids()))
|
||||||
|
self.hass.block_till_done()
|
||||||
|
|
||||||
|
state_1 = self.hass.states.get('counter.test_1')
|
||||||
|
state_2 = self.hass.states.get('counter.test_2')
|
||||||
|
|
||||||
|
self.assertIsNotNone(state_1)
|
||||||
|
self.assertIsNotNone(state_2)
|
||||||
|
|
||||||
|
self.assertEqual(0, int(state_1.state))
|
||||||
|
self.assertNotIn(ATTR_ICON, state_1.attributes)
|
||||||
|
self.assertNotIn(ATTR_FRIENDLY_NAME, state_1.attributes)
|
||||||
|
|
||||||
|
self.assertEqual(10, int(state_2.state))
|
||||||
|
self.assertEqual('Hello World',
|
||||||
|
state_2.attributes.get(ATTR_FRIENDLY_NAME))
|
||||||
|
self.assertEqual('mdi:work', state_2.attributes.get(ATTR_ICON))
|
||||||
|
|
||||||
|
|
||||||
|
@asyncio.coroutine
|
||||||
|
def test_initial_state_overrules_restore_state(hass):
|
||||||
|
"""Ensure states are restored on startup."""
|
||||||
|
mock_restore_cache(hass, (
|
||||||
|
State('counter.test1', '11'),
|
||||||
|
State('counter.test2', '-22'),
|
||||||
|
))
|
||||||
|
|
||||||
|
hass.state = CoreState.starting
|
||||||
|
|
||||||
|
yield from async_setup_component(hass, DOMAIN, {
|
||||||
|
DOMAIN: {
|
||||||
|
'test1': {},
|
||||||
|
'test2': {
|
||||||
|
CONF_INITIAL: 10,
|
||||||
|
},
|
||||||
|
}})
|
||||||
|
|
||||||
|
state = hass.states.get('counter.test1')
|
||||||
|
assert state
|
||||||
|
assert int(state.state) == 0
|
||||||
|
|
||||||
|
state = hass.states.get('counter.test2')
|
||||||
|
assert state
|
||||||
|
assert int(state.state) == 10
|
||||||
|
|
||||||
|
|
||||||
|
@asyncio.coroutine
|
||||||
|
def test_no_initial_state_and_no_restore_state(hass):
|
||||||
|
"""Ensure that entity is create without initial and restore feature."""
|
||||||
|
hass.state = CoreState.starting
|
||||||
|
|
||||||
|
yield from async_setup_component(hass, DOMAIN, {
|
||||||
|
DOMAIN: {
|
||||||
|
'test1': {
|
||||||
|
CONF_STEP: 5,
|
||||||
|
}
|
||||||
|
}})
|
||||||
|
|
||||||
|
state = hass.states.get('counter.test1')
|
||||||
|
assert state
|
||||||
|
assert int(state.state) == 0
|
Loading…
x
Reference in New Issue
Block a user