States
Home Assistant keeps track of the states of entities in a state machine. The state machine has very few requirements:
- Each state is related to an entity identified by an entity id. This id is made up of a domain and an object id. For example
light.kitchen_ceiling
. You can make up any combination of domain and object id, even overwriting existing states. - Each state has a primary attribute that describes the state of the entity. In the case of a light this could be for example "on" and "off". You can store anything you want in the state, as long as it's a string (will be converted if it's not).
- You can store more information about an entity by setting attributes. Attributes is a dictionary that can contain any data that you want. The only requirement is that it's JSON serializable, so you're limited to numbers, strings, dictionaries and lists.
Description of the state object.
Using states in your component
This is a simple tutorial/example on how to create and set states. We will do our work in a component called "hello_state". The purpose of this component is to display a given text in the frontend.
To get started, create the file <config dir>/custom_components/hello_state.py
and copy the below example code.
"""
Support for showing text in the frontend.
For more details about this component, please refer to the documentation at
https://developers.home-assistant.io/docs/dev_101_states
"""
import logging
_LOGGER = logging.getLogger(__name__)
DOMAIN = "hello_state"
def setup(hass, config):
"""Setup the Hello State component. """
_LOGGER.info("The 'hello state' component is ready!")
return True
In the file header we decided to add some details: A short description and the link to the documentation.
We want to do some logging. This means that we import the Python logging module and create an alias.
The component name is equal to the domain name.
The
setup
function will take care of the initialization of our component. The component will only write a log message. Keep in mind for later that you have several options for the severity:_LOGGER.info(msg)
_LOGGER.warning(msg)
_LOGGER.error(msg)
_LOGGER.critical(msg)
_LOGGER.exception(msg)
We return
True
if everything is ok.
Add the component to your configuration.yaml
file.
hello_state:
After a start or a restart of Home Assistant the component will create an entry in the log.
16-03-12 14:16:42 INFO (MainThread) [custom_components.hello_state] The 'hello state' component is ready!
The next step is the introduction of configuration options. A user can pass configuration options to our component via configuration.yaml
. To use them we'll use the passed in config
variable to our setup
method.
import logging
_LOGGER = logging.getLogger(__name__)
DOMAIN = "hello_state"
CONF_TEXT = "text"
DEFAULT_TEXT = "No text!"
def setup(hass, config):
"""Set up the Hello State component. """
# Get the text from the configuration. Use DEFAULT_TEXT if no name is provided.
text = config[DOMAIN].get(CONF_TEXT, DEFAULT_TEXT)
# States are in the format DOMAIN.OBJECT_ID
hass.states.set("hello_state.Hello_State", text)
return True
To use the latest feature of our component, update the entry in your configuration.yaml
file.
hello_state:
text: 'Hello, World!'
Thanks to DEFAULT_TEXT
variable the component will launch even if no text:
field is used in the configuration.yaml
file. Quite often there are variables which are required. It's important to check if all mandatory configuration variables are provided. If not, the setup should fail. We will use voluptuous
as a helper to achieve this. The next listing shows the essential parts.
import voluptuous as vol
import homeassistant.helpers.config_validation as cv
CONFIG_SCHEMA = vol.Schema(
{DOMAIN: vol.Schema({vol.Required(CONF_TEXT): cv.string,})}, extra=vol.ALLOW_EXTRA
)
Now, when text:
is missing from the config, Home Assistant will alert the user and not setup your component.
After a start or a restart of Home Assistant the component will be visible in the frontend if the configuration.yaml
file is up-to-date.
In order to expose attributes for a platform, you will need to define a property called extra_state_attributes
on the entity class, which will return a dictionary of attributes:
@property
def extra_state_attributes(self):
"""Return entity specific state attributes."""
return self._attributes
Entities also have a similar property state_attributes
, which should not be overridden by integrations. This property is used by base entity components to add standard sets of attributes to a state. Example: The light component uses state_attributes
to add brightness to the state dictionary. If you are designing a new integration, you should define extra_state_attributes
instead.
To get your integration included in the Home Assistant releases, follow the steps described in the Submit your work section. Basically you only need to move your integration into the homeassistant/component/
directory of your fork and create a Pull Request.