mirror of
https://github.com/home-assistant/core.git
synced 2025-09-21 19:09:35 +00:00
Compare commits
31 Commits
Author | SHA1 | Date | |
---|---|---|---|
![]() |
0323aeb660 | ||
![]() |
58e8162d18 | ||
![]() |
8036b709e0 | ||
![]() |
e80ffe16f7 | ||
![]() |
99010bab18 | ||
![]() |
8d5e15df3d | ||
![]() |
efffc82414 | ||
![]() |
06b99a1ff9 | ||
![]() |
fd79c141b2 | ||
![]() |
b541b08d2d | ||
![]() |
42c9776c97 | ||
![]() |
897340b115 | ||
![]() |
e072d8b694 | ||
![]() |
52d8c66a8b | ||
![]() |
a26413c989 | ||
![]() |
c9d53dec8d | ||
![]() |
dae2117078 | ||
![]() |
72b95f7529 | ||
![]() |
56436e6387 | ||
![]() |
7cdbbc248f | ||
![]() |
cca2594225 | ||
![]() |
22bef1d0d3 | ||
![]() |
b2885f6cc5 | ||
![]() |
3a1ee2f654 | ||
![]() |
26ba956242 | ||
![]() |
90d8874825 | ||
![]() |
22f76a9363 | ||
![]() |
cc4d71f942 | ||
![]() |
0a55f024a4 | ||
![]() |
60fe64d119 | ||
![]() |
a3692859e9 |
@@ -115,10 +115,10 @@ stages:
|
||||
docker run --rm --privileged \
|
||||
-v ~/.docker:/root/.docker \
|
||||
-v /run/docker.sock:/run/docker.sock:rw \
|
||||
-v $(pwd):/data:ro \
|
||||
homeassistant/amd64-builder:$(versionBuilder) \
|
||||
--homeassistant-machine "$(homeassistantRelease)=$(buildMachine)" \
|
||||
-r https://github.com/home-assistant/hassio-homeassistant \
|
||||
-t machine --docker-hub homeassistant
|
||||
-t /data/machine --docker-hub homeassistant
|
||||
displayName: 'Build Release'
|
||||
|
||||
- stage: 'Publish'
|
||||
|
@@ -3,7 +3,7 @@
|
||||
"name": "Arris TG2492LG",
|
||||
"documentation": "https://www.home-assistant.io/integrations/arris_tg2492lg",
|
||||
"requirements": [
|
||||
"arris-tg2492lg==1.0.0"
|
||||
"arris-tg2492lg==1.1.0"
|
||||
],
|
||||
"codeowners": [
|
||||
"@vanbalken"
|
||||
|
@@ -1,4 +1,5 @@
|
||||
"""The devolo_home_control integration."""
|
||||
import asyncio
|
||||
from functools import partial
|
||||
|
||||
from devolo_home_control_api.homecontrol import HomeControl
|
||||
@@ -71,8 +72,13 @@ async def async_setup_entry(hass: HomeAssistantType, entry: ConfigEntry) -> bool
|
||||
|
||||
async def async_unload_entry(hass, config_entry):
|
||||
"""Unload a config entry."""
|
||||
unload = await hass.config_entries.async_forward_entry_unload(
|
||||
config_entry, "switch"
|
||||
unload = all(
|
||||
await asyncio.gather(
|
||||
*[
|
||||
hass.config_entries.async_forward_entry_unload(config_entry, platform)
|
||||
for platform in PLATFORMS
|
||||
]
|
||||
)
|
||||
)
|
||||
|
||||
await hass.async_add_executor_job(
|
||||
|
@@ -2,7 +2,7 @@
|
||||
"domain": "frontend",
|
||||
"name": "Home Assistant Frontend",
|
||||
"documentation": "https://www.home-assistant.io/integrations/frontend",
|
||||
"requirements": ["home-assistant-frontend==20201001.1"],
|
||||
"requirements": ["home-assistant-frontend==20201001.2"],
|
||||
"dependencies": [
|
||||
"api",
|
||||
"auth",
|
||||
|
@@ -65,6 +65,7 @@ class HassIOView(HomeAssistantView):
|
||||
|
||||
return await self._command_proxy(path, request)
|
||||
|
||||
delete = _handle
|
||||
get = _handle
|
||||
post = _handle
|
||||
|
||||
|
@@ -2,6 +2,6 @@
|
||||
"domain": "hive",
|
||||
"name": "Hive",
|
||||
"documentation": "https://www.home-assistant.io/integrations/hive",
|
||||
"requirements": ["pyhiveapi==0.2.20.1"],
|
||||
"requirements": ["pyhiveapi==0.2.20.2"],
|
||||
"codeowners": ["@Rendili", "@KJonline"]
|
||||
}
|
||||
|
@@ -3,5 +3,5 @@
|
||||
"name": "KEF",
|
||||
"documentation": "https://www.home-assistant.io/integrations/kef",
|
||||
"codeowners": ["@basnijholt"],
|
||||
"requirements": ["aiokef==0.2.13", "getmac==0.8.2"]
|
||||
"requirements": ["aiokef==0.2.16", "getmac==0.8.2"]
|
||||
}
|
||||
|
@@ -1,6 +1,5 @@
|
||||
"""Platform for the KEF Wireless Speakers."""
|
||||
|
||||
import asyncio
|
||||
from datetime import timedelta
|
||||
from functools import partial
|
||||
import ipaddress
|
||||
@@ -164,7 +163,11 @@ async def async_setup_platform(hass, config, async_add_entities, discovery_info=
|
||||
dtype = type(options[0]) # int or float
|
||||
platform.async_register_entity_service(
|
||||
name,
|
||||
{vol.Required(option): vol.All(vol.Coerce(dtype), vol.In(options))},
|
||||
{
|
||||
vol.Required(option): vol.All(
|
||||
vol.Coerce(float), vol.Coerce(dtype), vol.In(options)
|
||||
)
|
||||
},
|
||||
f"set_{which}",
|
||||
)
|
||||
|
||||
@@ -365,17 +368,16 @@ class KefMediaPlayer(MediaPlayerEntity):
|
||||
# The LSX is able to respond when off the LS50 has to be on.
|
||||
return
|
||||
|
||||
(mode, *rest) = await asyncio.gather(
|
||||
self._speaker.get_mode(),
|
||||
self._speaker.get_desk_db(),
|
||||
self._speaker.get_wall_db(),
|
||||
self._speaker.get_treble_db(),
|
||||
self._speaker.get_high_hz(),
|
||||
self._speaker.get_low_hz(),
|
||||
self._speaker.get_sub_db(),
|
||||
mode = await self._speaker.get_mode()
|
||||
self._dsp = dict(
|
||||
desk_db=await self._speaker.get_desk_db(),
|
||||
wall_db=await self._speaker.get_wall_db(),
|
||||
treble_db=await self._speaker.get_treble_db(),
|
||||
high_hz=await self._speaker.get_high_hz(),
|
||||
low_hz=await self._speaker.get_low_hz(),
|
||||
sub_db=await self._speaker.get_sub_db(),
|
||||
**mode._asdict(),
|
||||
)
|
||||
keys = ["desk_db", "wall_db", "treble_db", "high_hz", "low_hz", "sub_db"]
|
||||
self._dsp = dict(zip(keys, rest), **mode._asdict())
|
||||
|
||||
async def async_added_to_hass(self):
|
||||
"""Subscribe to DSP updates."""
|
||||
|
@@ -2,16 +2,22 @@
|
||||
|
||||
|
||||
from homeassistant.components.group import GroupIntegrationRegistry
|
||||
from homeassistant.const import STATE_OFF
|
||||
from homeassistant.const import (
|
||||
STATE_IDLE,
|
||||
STATE_OFF,
|
||||
STATE_ON,
|
||||
STATE_PAUSED,
|
||||
STATE_PLAYING,
|
||||
)
|
||||
from homeassistant.core import callback
|
||||
from homeassistant.helpers.typing import HomeAssistantType
|
||||
|
||||
from . import STATE_IDLE, STATE_PLAYING
|
||||
|
||||
|
||||
@callback
|
||||
def async_describe_on_off_states(
|
||||
hass: HomeAssistantType, registry: GroupIntegrationRegistry
|
||||
) -> None:
|
||||
"""Describe group on off states."""
|
||||
registry.on_off_states({STATE_PLAYING, STATE_IDLE}, STATE_OFF)
|
||||
registry.on_off_states(
|
||||
{STATE_ON, STATE_PAUSED, STATE_PLAYING, STATE_IDLE}, STATE_OFF
|
||||
)
|
||||
|
@@ -3,7 +3,7 @@
|
||||
"name": "MQTT",
|
||||
"config_flow": true,
|
||||
"documentation": "https://www.home-assistant.io/integrations/mqtt",
|
||||
"requirements": ["paho-mqtt==1.5.1"],
|
||||
"requirements": ["paho-mqtt==1.5.0"],
|
||||
"dependencies": ["http"],
|
||||
"codeowners": ["@home-assistant/core", "@emontnemery"]
|
||||
}
|
||||
|
@@ -41,7 +41,7 @@ DEFAULT_INTERVALS = {
|
||||
HOMEDATA_DATA_CLASS_NAME: 900,
|
||||
HOMESTATUS_DATA_CLASS_NAME: 300,
|
||||
CAMERA_DATA_CLASS_NAME: 900,
|
||||
WEATHERSTATION_DATA_CLASS_NAME: 300,
|
||||
WEATHERSTATION_DATA_CLASS_NAME: 600,
|
||||
HOMECOACH_DATA_CLASS_NAME: 300,
|
||||
PUBLICDATA_DATA_CLASS_NAME: 600,
|
||||
}
|
||||
|
@@ -3,7 +3,7 @@
|
||||
"name": "Netatmo",
|
||||
"documentation": "https://www.home-assistant.io/integrations/netatmo",
|
||||
"requirements": [
|
||||
"pyatmo==4.0.0"
|
||||
"pyatmo==4.1.0"
|
||||
],
|
||||
"after_dependencies": [
|
||||
"cloud",
|
||||
|
@@ -136,6 +136,7 @@ async def async_setup_entry(hass, entry, async_add_entities):
|
||||
conditions = [
|
||||
c.lower()
|
||||
for c in data_class.get_monitored_conditions(module_id=module["_id"])
|
||||
if c.lower() in SENSOR_TYPES
|
||||
]
|
||||
for condition in conditions:
|
||||
if f"{condition}_value" in SENSOR_TYPES:
|
||||
|
@@ -119,7 +119,7 @@ class OmniLogicTemperatureSensor(OmnilogicSensor):
|
||||
state = sensor_data
|
||||
|
||||
if self._unit_type == "Metric":
|
||||
hayward_state = round((hayward_state - 32) * 5 / 9, 1)
|
||||
hayward_state = round((int(hayward_state) - 32) * 5 / 9, 1)
|
||||
hayward_unit_of_measure = TEMP_CELSIUS
|
||||
|
||||
if int(sensor_data) == -1:
|
||||
@@ -175,7 +175,7 @@ class OmniLogicSaltLevelSensor(OmnilogicSensor):
|
||||
unit_of_measurement = self._unit
|
||||
|
||||
if self._unit_type == "Metric":
|
||||
salt_return = round(salt_return / 1000, 2)
|
||||
salt_return = round(int(salt_return) / 1000, 2)
|
||||
unit_of_measurement = f"{MASS_GRAMS}/{VOLUME_LITERS}"
|
||||
|
||||
self._unit = unit_of_measurement
|
||||
@@ -279,7 +279,7 @@ SENSOR_TYPES = {
|
||||
"icon": "mdi:speedometer",
|
||||
"unit": PERCENTAGE,
|
||||
"guard_condition": [
|
||||
{"Type": "FMT_SINGLE_SPEED"},
|
||||
{"Filter-Type": "FMT_SINGLE_SPEED"},
|
||||
],
|
||||
},
|
||||
],
|
||||
|
15
homeassistant/components/plant/group.py
Normal file
15
homeassistant/components/plant/group.py
Normal file
@@ -0,0 +1,15 @@
|
||||
"""Describe group states."""
|
||||
|
||||
|
||||
from homeassistant.components.group import GroupIntegrationRegistry
|
||||
from homeassistant.const import STATE_OK, STATE_PROBLEM
|
||||
from homeassistant.core import callback
|
||||
from homeassistant.helpers.typing import HomeAssistantType
|
||||
|
||||
|
||||
@callback
|
||||
def async_describe_on_off_states(
|
||||
hass: HomeAssistantType, registry: GroupIntegrationRegistry
|
||||
) -> None:
|
||||
"""Describe group on off states."""
|
||||
registry.on_off_states({STATE_PROBLEM}, STATE_OK)
|
@@ -3,7 +3,7 @@
|
||||
"step": {
|
||||
"user": {
|
||||
"title": "PoolSense",
|
||||
"description": "[%key:common::config_flow::description%]",
|
||||
"description": "[%key:common::config_flow::description::confirm_setup%]",
|
||||
"data": {
|
||||
"email": "[%key:common::config_flow::data::email%]",
|
||||
"password": "[%key:common::config_flow::data::password%]"
|
||||
|
@@ -494,7 +494,8 @@ class Recorder(threading.Thread):
|
||||
for dbstate in self._pending_expunge:
|
||||
# Expunge the state so its not expired
|
||||
# until we use it later for dbstate.old_state
|
||||
self.event_session.expunge(dbstate)
|
||||
if dbstate in self.event_session:
|
||||
self.event_session.expunge(dbstate)
|
||||
self._pending_expunge = []
|
||||
self.event_session.commit()
|
||||
except Exception as err:
|
||||
|
@@ -28,7 +28,7 @@ def shelly_naming(self, block, entity_type: str):
|
||||
return f"{entity_name} {self.description.name}"
|
||||
|
||||
channels = 0
|
||||
mode = "relays"
|
||||
mode = block.type + "s"
|
||||
if "num_outputs" in self.wrapper.device.shelly:
|
||||
channels = self.wrapper.device.shelly["num_outputs"]
|
||||
if (
|
||||
@@ -38,7 +38,6 @@ def shelly_naming(self, block, entity_type: str):
|
||||
channels = 1
|
||||
if block.type == "emeter" and "num_emeters" in self.wrapper.device.shelly:
|
||||
channels = self.wrapper.device.shelly["num_emeters"]
|
||||
mode = "emeters"
|
||||
if channels > 1 and block.type != "device":
|
||||
# Shelly EM (SHEM) with firmware v1.8.1 doesn't have "name" key; will be fixed in next firmware release
|
||||
if "name" in self.wrapper.device.settings[mode][int(block.channel)]:
|
||||
|
@@ -16,7 +16,7 @@ from simplipy.websocket import (
|
||||
)
|
||||
import voluptuous as vol
|
||||
|
||||
from homeassistant.config_entries import SOURCE_IMPORT
|
||||
from homeassistant.config_entries import SOURCE_IMPORT, SOURCE_REAUTH
|
||||
from homeassistant.const import (
|
||||
ATTR_CODE,
|
||||
CONF_CODE,
|
||||
@@ -365,8 +365,7 @@ async def async_unload_entry(hass, entry):
|
||||
|
||||
async def async_update_options(hass, config_entry):
|
||||
"""Handle an options update."""
|
||||
simplisafe = hass.data[DOMAIN][DATA_CLIENT][config_entry.entry_id]
|
||||
simplisafe.options = config_entry.options
|
||||
await hass.config_entries.async_reload(config_entry.entry_id)
|
||||
|
||||
|
||||
class SimpliSafeWebsocket:
|
||||
@@ -530,17 +529,26 @@ class SimpliSafe:
|
||||
for result in results:
|
||||
if isinstance(result, InvalidCredentialsError):
|
||||
if self._emergency_refresh_token_used:
|
||||
LOGGER.error(
|
||||
"Token disconnected or invalid. Please re-auth the "
|
||||
"SimpliSafe integration in HASS"
|
||||
)
|
||||
self._hass.async_create_task(
|
||||
self._hass.config_entries.flow.async_init(
|
||||
DOMAIN,
|
||||
context={"source": "reauth"},
|
||||
data=self._config_entry.data,
|
||||
matching_flows = [
|
||||
flow
|
||||
for flow in self._hass.config_entries.flow.async_progress()
|
||||
if flow["context"].get("source") == SOURCE_REAUTH
|
||||
and flow["context"].get("unique_id")
|
||||
== self._config_entry.unique_id
|
||||
]
|
||||
|
||||
if not matching_flows:
|
||||
self._hass.async_create_task(
|
||||
self._hass.config_entries.flow.async_init(
|
||||
DOMAIN,
|
||||
context={
|
||||
"source": SOURCE_REAUTH,
|
||||
"unique_id": self._config_entry.unique_id,
|
||||
},
|
||||
data=self._config_entry.data,
|
||||
)
|
||||
)
|
||||
)
|
||||
|
||||
return
|
||||
|
||||
LOGGER.warning("SimpliSafe cloud error; trying stored refresh token")
|
||||
|
@@ -2,12 +2,13 @@
|
||||
from typing import Dict
|
||||
|
||||
from synology_dsm.api.surveillance_station import SynoSurveillanceStation
|
||||
from synology_dsm.api.surveillance_station.camera import SynoCamera
|
||||
|
||||
from homeassistant.components.camera import SUPPORT_STREAM, Camera
|
||||
from homeassistant.config_entries import ConfigEntry
|
||||
from homeassistant.helpers.typing import HomeAssistantType
|
||||
|
||||
from . import SynologyDSMEntity
|
||||
from . import SynoApi, SynologyDSMEntity
|
||||
from .const import (
|
||||
DOMAIN,
|
||||
ENTITY_CLASS,
|
||||
@@ -40,7 +41,7 @@ async def async_setup_entry(
|
||||
class SynoDSMCamera(SynologyDSMEntity, Camera):
|
||||
"""Representation a Synology camera."""
|
||||
|
||||
def __init__(self, api, camera):
|
||||
def __init__(self, api: SynoApi, camera: SynoCamera):
|
||||
"""Initialize a Synology camera."""
|
||||
super().__init__(
|
||||
api,
|
||||
@@ -69,6 +70,11 @@ class SynoDSMCamera(SynologyDSMEntity, Camera):
|
||||
),
|
||||
}
|
||||
|
||||
@property
|
||||
def available(self) -> bool:
|
||||
"""Return the availability of the camera."""
|
||||
return self._camera.is_enabled
|
||||
|
||||
@property
|
||||
def supported_features(self) -> int:
|
||||
"""Return supported features of this camera."""
|
||||
@@ -86,10 +92,14 @@ class SynoDSMCamera(SynologyDSMEntity, Camera):
|
||||
|
||||
def camera_image(self) -> bytes:
|
||||
"""Return bytes of camera image."""
|
||||
if not self.available:
|
||||
return None
|
||||
return self._api.surveillance_station.get_camera_image(self._camera.id)
|
||||
|
||||
async def stream_source(self) -> str:
|
||||
"""Return the source of the stream."""
|
||||
if not self.available:
|
||||
return None
|
||||
return self._camera.live_view.rtsp
|
||||
|
||||
def enable_motion_detection(self):
|
||||
|
@@ -257,13 +257,20 @@ async def handle_render_template(hass, connection, msg):
|
||||
timeout = msg.get("timeout")
|
||||
info = None
|
||||
|
||||
if timeout and await template.async_render_will_timeout(timeout):
|
||||
connection.send_error(
|
||||
msg["id"],
|
||||
const.ERR_TEMPLATE_ERROR,
|
||||
f"Exceeded maximum execution time of {timeout}s",
|
||||
)
|
||||
return
|
||||
if timeout:
|
||||
try:
|
||||
timed_out = await template.async_render_will_timeout(timeout)
|
||||
except TemplateError as ex:
|
||||
connection.send_error(msg["id"], const.ERR_TEMPLATE_ERROR, str(ex))
|
||||
return
|
||||
|
||||
if timed_out:
|
||||
connection.send_error(
|
||||
msg["id"],
|
||||
const.ERR_TEMPLATE_ERROR,
|
||||
f"Exceeded maximum execution time of {timeout}s",
|
||||
)
|
||||
return
|
||||
|
||||
@callback
|
||||
def _template_listener(event, updates):
|
||||
|
@@ -137,7 +137,7 @@ EFFECTS_MAP = {
|
||||
EFFECT_POLICE2: yee_transitions.police2,
|
||||
EFFECT_CHRISTMAS: yee_transitions.christmas,
|
||||
EFFECT_RGB: yee_transitions.rgb,
|
||||
EFFECT_RANDOM_LOOP: yee_transitions.randomloop,
|
||||
EFFECT_RANDOM_LOOP: yee_transitions.random_loop,
|
||||
EFFECT_LSD: yee_transitions.lsd,
|
||||
EFFECT_SLOWDOWN: yee_transitions.slowdown,
|
||||
}
|
||||
@@ -661,7 +661,7 @@ class YeelightGenericLight(YeelightEntity, LightEntity):
|
||||
elif effect in EFFECTS_MAP:
|
||||
flow = Flow(count=0, transitions=EFFECTS_MAP[effect]())
|
||||
elif effect == EFFECT_FAST_RANDOM_LOOP:
|
||||
flow = Flow(count=0, transitions=yee_transitions.randomloop(duration=250))
|
||||
flow = Flow(count=0, transitions=yee_transitions.random_loop(duration=250))
|
||||
elif effect == EFFECT_WHATSAPP:
|
||||
flow = Flow(count=2, transitions=yee_transitions.pulse(37, 211, 102))
|
||||
elif effect == EFFECT_FACEBOOK:
|
||||
|
@@ -3,7 +3,7 @@
|
||||
"name": "Yeelight",
|
||||
"documentation": "https://www.home-assistant.io/integrations/yeelight",
|
||||
"requirements": [
|
||||
"yeelight==0.5.3"
|
||||
"yeelight==0.5.4"
|
||||
],
|
||||
"codeowners": [
|
||||
"@rytilahti",
|
||||
|
@@ -1,7 +1,7 @@
|
||||
"""Constants used by Home Assistant components."""
|
||||
MAJOR_VERSION = 0
|
||||
MINOR_VERSION = 116
|
||||
PATCH_VERSION = "0"
|
||||
PATCH_VERSION = "3"
|
||||
__short_version__ = f"{MAJOR_VERSION}.{MINOR_VERSION}"
|
||||
__version__ = f"{__short_version__}.{PATCH_VERSION}"
|
||||
REQUIRED_PYTHON_VER = (3, 7, 1)
|
||||
|
@@ -1,8 +1,6 @@
|
||||
"""The exceptions used by Home Assistant."""
|
||||
from typing import TYPE_CHECKING, Optional
|
||||
|
||||
import jinja2
|
||||
|
||||
if TYPE_CHECKING:
|
||||
from .core import Context # noqa: F401 pylint: disable=unused-import
|
||||
|
||||
@@ -22,7 +20,7 @@ class NoEntitySpecifiedError(HomeAssistantError):
|
||||
class TemplateError(HomeAssistantError):
|
||||
"""Error during template rendering."""
|
||||
|
||||
def __init__(self, exception: jinja2.TemplateError) -> None:
|
||||
def __init__(self, exception: Exception) -> None:
|
||||
"""Init the error."""
|
||||
super().__init__(f"{exception.__class__.__name__}: {exception}")
|
||||
|
||||
|
@@ -233,6 +233,9 @@ def async_track_state_change_event(
|
||||
care about the state change events so we can
|
||||
do a fast dict lookup to route events.
|
||||
"""
|
||||
entity_ids = _async_string_to_lower_list(entity_ids)
|
||||
if not entity_ids:
|
||||
return _remove_empty_listener
|
||||
|
||||
entity_callbacks = hass.data.setdefault(TRACK_STATE_CHANGE_CALLBACKS, {})
|
||||
|
||||
@@ -277,6 +280,11 @@ def async_track_state_change_event(
|
||||
return remove_listener
|
||||
|
||||
|
||||
@callback
|
||||
def _remove_empty_listener() -> None:
|
||||
"""Remove a listener that does nothing."""
|
||||
|
||||
|
||||
@callback
|
||||
def _async_remove_indexed_listeners(
|
||||
hass: HomeAssistant,
|
||||
@@ -309,6 +317,9 @@ def async_track_entity_registry_updated_event(
|
||||
|
||||
Similar to async_track_state_change_event.
|
||||
"""
|
||||
entity_ids = _async_string_to_lower_list(entity_ids)
|
||||
if not entity_ids:
|
||||
return _remove_empty_listener
|
||||
|
||||
entity_callbacks = hass.data.setdefault(TRACK_ENTITY_REGISTRY_UPDATED_CALLBACKS, {})
|
||||
|
||||
@@ -381,6 +392,9 @@ def async_track_state_added_domain(
|
||||
action: Callable[[Event], Any],
|
||||
) -> Callable[[], None]:
|
||||
"""Track state change events when an entity is added to domains."""
|
||||
domains = _async_string_to_lower_list(domains)
|
||||
if not domains:
|
||||
return _remove_empty_listener
|
||||
|
||||
domain_callbacks = hass.data.setdefault(TRACK_STATE_ADDED_DOMAIN_CALLBACKS, {})
|
||||
|
||||
@@ -424,6 +438,9 @@ def async_track_state_removed_domain(
|
||||
action: Callable[[Event], Any],
|
||||
) -> Callable[[], None]:
|
||||
"""Track state change events when an entity is removed from domains."""
|
||||
domains = _async_string_to_lower_list(domains)
|
||||
if not domains:
|
||||
return _remove_empty_listener
|
||||
|
||||
domain_callbacks = hass.data.setdefault(TRACK_STATE_REMOVED_DOMAIN_CALLBACKS, {})
|
||||
|
||||
|
@@ -330,7 +330,7 @@ class Template:
|
||||
|
||||
try:
|
||||
return compiled.render(kwargs).strip()
|
||||
except jinja2.TemplateError as err:
|
||||
except Exception as err: # pylint: disable=broad-except
|
||||
raise TemplateError(err) from err
|
||||
|
||||
async def async_render_will_timeout(
|
||||
|
@@ -13,11 +13,11 @@ defusedxml==0.6.0
|
||||
distro==1.5.0
|
||||
emoji==0.5.4
|
||||
hass-nabucasa==0.37.0
|
||||
home-assistant-frontend==20201001.1
|
||||
home-assistant-frontend==20201001.2
|
||||
importlib-metadata==1.6.0;python_version<'3.8'
|
||||
jinja2>=2.11.2
|
||||
netdisco==2.8.2
|
||||
paho-mqtt==1.5.1
|
||||
paho-mqtt==1.5.0
|
||||
pillow==7.2.0
|
||||
pip>=8.0.3
|
||||
python-slugify==4.0.1
|
||||
|
34
machine/intel-nuc
Normal file
34
machine/intel-nuc
Normal file
@@ -0,0 +1,34 @@
|
||||
ARG BUILD_VERSION
|
||||
FROM homeassistant/amd64-homeassistant:$BUILD_VERSION
|
||||
|
||||
RUN apk --no-cache add \
|
||||
libva-intel-driver \
|
||||
usbutils
|
||||
|
||||
##
|
||||
# Build libcec for HDMI-CEC
|
||||
ARG LIBCEC_VERSION=6.0.2
|
||||
RUN apk add --no-cache \
|
||||
eudev-libs \
|
||||
p8-platform \
|
||||
&& apk add --no-cache --virtual .build-dependencies \
|
||||
build-base \
|
||||
cmake \
|
||||
eudev-dev \
|
||||
swig \
|
||||
p8-platform-dev \
|
||||
linux-headers \
|
||||
&& git clone --depth 1 -b libcec-${LIBCEC_VERSION} https://github.com/Pulse-Eight/libcec /usr/src/libcec \
|
||||
&& cd /usr/src/libcec \
|
||||
&& mkdir -p /usr/src/libcec/build \
|
||||
&& cd /usr/src/libcec/build \
|
||||
&& cmake -DCMAKE_INSTALL_PREFIX:PATH=/usr/local \
|
||||
-DPYTHON_LIBRARY="/usr/local/lib/libpython3.8.so" \
|
||||
-DPYTHON_INCLUDE_DIR="/usr/local/include/python3.8" \
|
||||
-DHAVE_LINUX_API=1 \
|
||||
.. \
|
||||
&& make -j$(nproc) \
|
||||
&& make install \
|
||||
&& echo "cec" > "/usr/local/lib/python3.8/site-packages/cec.pth" \
|
||||
&& apk del .build-dependencies \
|
||||
&& rm -rf /usr/src/libcec*
|
34
machine/odroid-c2
Normal file
34
machine/odroid-c2
Normal file
@@ -0,0 +1,34 @@
|
||||
ARG BUILD_VERSION
|
||||
FROM homeassistant/aarch64-homeassistant:$BUILD_VERSION
|
||||
|
||||
RUN apk --no-cache add \
|
||||
usbutils
|
||||
|
||||
##
|
||||
# Build libcec for HDMI-CEC
|
||||
ARG LIBCEC_VERSION=6.0.2
|
||||
RUN apk add --no-cache \
|
||||
eudev-libs \
|
||||
p8-platform \
|
||||
&& apk add --no-cache --virtual .build-dependencies \
|
||||
build-base \
|
||||
cmake \
|
||||
eudev-dev \
|
||||
swig \
|
||||
p8-platform-dev \
|
||||
linux-headers \
|
||||
&& git clone --depth 1 -b libcec-${LIBCEC_VERSION} https://github.com/Pulse-Eight/libcec /usr/src/libcec \
|
||||
&& cd /usr/src/libcec \
|
||||
&& mkdir -p /usr/src/libcec/build \
|
||||
&& cd /usr/src/libcec/build \
|
||||
&& cmake -DCMAKE_INSTALL_PREFIX:PATH=/usr/local \
|
||||
-DPYTHON_LIBRARY="/usr/local/lib/libpython3.8.so" \
|
||||
-DPYTHON_INCLUDE_DIR="/usr/local/include/python3.8" \
|
||||
-DHAVE_LINUX_API=1 \
|
||||
-DHAVE_AOCEC_API=1 \
|
||||
.. \
|
||||
&& make -j$(nproc) \
|
||||
&& make install \
|
||||
&& echo "cec" > "/usr/local/lib/python3.8/site-packages/cec.pth" \
|
||||
&& apk del .build-dependencies \
|
||||
&& rm -rf /usr/src/libcec*
|
34
machine/odroid-n2
Normal file
34
machine/odroid-n2
Normal file
@@ -0,0 +1,34 @@
|
||||
ARG BUILD_VERSION
|
||||
FROM homeassistant/aarch64-homeassistant:$BUILD_VERSION
|
||||
|
||||
RUN apk --no-cache add \
|
||||
usbutils
|
||||
|
||||
##
|
||||
# Build libcec for HDMI-CEC
|
||||
ARG LIBCEC_VERSION=6.0.2
|
||||
RUN apk add --no-cache \
|
||||
eudev-libs \
|
||||
p8-platform \
|
||||
&& apk add --no-cache --virtual .build-dependencies \
|
||||
build-base \
|
||||
cmake \
|
||||
eudev-dev \
|
||||
swig \
|
||||
p8-platform-dev \
|
||||
linux-headers \
|
||||
&& git clone --depth 1 -b libcec-${LIBCEC_VERSION} https://github.com/Pulse-Eight/libcec /usr/src/libcec \
|
||||
&& cd /usr/src/libcec \
|
||||
&& mkdir -p /usr/src/libcec/build \
|
||||
&& cd /usr/src/libcec/build \
|
||||
&& cmake -DCMAKE_INSTALL_PREFIX:PATH=/usr/local \
|
||||
-DPYTHON_LIBRARY="/usr/local/lib/libpython3.8.so" \
|
||||
-DPYTHON_INCLUDE_DIR="/usr/local/include/python3.8" \
|
||||
-DHAVE_LINUX_API=1 \
|
||||
-DHAVE_AOCEC_API=1 \
|
||||
.. \
|
||||
&& make -j$(nproc) \
|
||||
&& make install \
|
||||
&& echo "cec" > "/usr/local/lib/python3.8/site-packages/cec.pth" \
|
||||
&& apk del .build-dependencies \
|
||||
&& rm -rf /usr/src/libcec*
|
34
machine/odroid-xu
Normal file
34
machine/odroid-xu
Normal file
@@ -0,0 +1,34 @@
|
||||
ARG BUILD_VERSION
|
||||
FROM homeassistant/armv7-homeassistant:$BUILD_VERSION
|
||||
|
||||
RUN apk --no-cache add \
|
||||
usbutils
|
||||
|
||||
##
|
||||
# Build libcec for HDMI-CEC
|
||||
ARG LIBCEC_VERSION=6.0.2
|
||||
RUN apk add --no-cache \
|
||||
eudev-libs \
|
||||
p8-platform \
|
||||
&& apk add --no-cache --virtual .build-dependencies \
|
||||
build-base \
|
||||
cmake \
|
||||
eudev-dev \
|
||||
swig \
|
||||
p8-platform-dev \
|
||||
linux-headers \
|
||||
&& git clone --depth 1 -b libcec-${LIBCEC_VERSION} https://github.com/Pulse-Eight/libcec /usr/src/libcec \
|
||||
&& cd /usr/src/libcec \
|
||||
&& mkdir -p /usr/src/libcec/build \
|
||||
&& cd /usr/src/libcec/build \
|
||||
&& cmake -DCMAKE_INSTALL_PREFIX:PATH=/usr/local \
|
||||
-DPYTHON_LIBRARY="/usr/local/lib/libpython3.8.so" \
|
||||
-DPYTHON_INCLUDE_DIR="/usr/local/include/python3.8" \
|
||||
-DHAVE_LINUX_API=1 \
|
||||
-DHAVE_EXYNOS_API=1 \
|
||||
.. \
|
||||
&& make -j$(nproc) \
|
||||
&& make install \
|
||||
&& echo "cec" > "/usr/local/lib/python3.8/site-packages/cec.pth" \
|
||||
&& apk del .build-dependencies \
|
||||
&& rm -rf /usr/src/libcec*
|
33
machine/qemuarm
Normal file
33
machine/qemuarm
Normal file
@@ -0,0 +1,33 @@
|
||||
ARG BUILD_VERSION
|
||||
FROM homeassistant/armhf-homeassistant:$BUILD_VERSION
|
||||
|
||||
RUN apk --no-cache add \
|
||||
usbutils
|
||||
|
||||
##
|
||||
# Build libcec for HDMI-CEC
|
||||
ARG LIBCEC_VERSION=6.0.2
|
||||
RUN apk add --no-cache \
|
||||
eudev-libs \
|
||||
p8-platform \
|
||||
&& apk add --no-cache --virtual .build-dependencies \
|
||||
build-base \
|
||||
cmake \
|
||||
eudev-dev \
|
||||
swig \
|
||||
p8-platform-dev \
|
||||
linux-headers \
|
||||
&& git clone --depth 1 -b libcec-${LIBCEC_VERSION} https://github.com/Pulse-Eight/libcec /usr/src/libcec \
|
||||
&& cd /usr/src/libcec \
|
||||
&& mkdir -p /usr/src/libcec/build \
|
||||
&& cd /usr/src/libcec/build \
|
||||
&& cmake -DCMAKE_INSTALL_PREFIX:PATH=/usr/local \
|
||||
-DPYTHON_LIBRARY="/usr/local/lib/libpython3.8.so" \
|
||||
-DPYTHON_INCLUDE_DIR="/usr/local/include/python3.8" \
|
||||
-DHAVE_LINUX_API=1 \
|
||||
.. \
|
||||
&& make -j$(nproc) \
|
||||
&& make install \
|
||||
&& echo "cec" > "/usr/local/lib/python3.8/site-packages/cec.pth" \
|
||||
&& apk del .build-dependencies \
|
||||
&& rm -rf /usr/src/libcec*
|
33
machine/qemuarm-64
Normal file
33
machine/qemuarm-64
Normal file
@@ -0,0 +1,33 @@
|
||||
ARG BUILD_VERSION
|
||||
FROM homeassistant/aarch64-homeassistant:$BUILD_VERSION
|
||||
|
||||
RUN apk --no-cache add \
|
||||
usbutils
|
||||
|
||||
##
|
||||
# Build libcec for HDMI-CEC
|
||||
ARG LIBCEC_VERSION=6.0.2
|
||||
RUN apk add --no-cache \
|
||||
eudev-libs \
|
||||
p8-platform \
|
||||
&& apk add --no-cache --virtual .build-dependencies \
|
||||
build-base \
|
||||
cmake \
|
||||
eudev-dev \
|
||||
swig \
|
||||
p8-platform-dev \
|
||||
linux-headers \
|
||||
&& git clone --depth 1 -b libcec-${LIBCEC_VERSION} https://github.com/Pulse-Eight/libcec /usr/src/libcec \
|
||||
&& cd /usr/src/libcec \
|
||||
&& mkdir -p /usr/src/libcec/build \
|
||||
&& cd /usr/src/libcec/build \
|
||||
&& cmake -DCMAKE_INSTALL_PREFIX:PATH=/usr/local \
|
||||
-DPYTHON_LIBRARY="/usr/local/lib/libpython3.8.so" \
|
||||
-DPYTHON_INCLUDE_DIR="/usr/local/include/python3.8" \
|
||||
-DHAVE_LINUX_API=1 \
|
||||
.. \
|
||||
&& make -j$(nproc) \
|
||||
&& make install \
|
||||
&& echo "cec" > "/usr/local/lib/python3.8/site-packages/cec.pth" \
|
||||
&& apk del .build-dependencies \
|
||||
&& rm -rf /usr/src/libcec*
|
33
machine/qemux86
Normal file
33
machine/qemux86
Normal file
@@ -0,0 +1,33 @@
|
||||
ARG BUILD_VERSION
|
||||
FROM homeassistant/i386-homeassistant:$BUILD_VERSION
|
||||
|
||||
RUN apk --no-cache add \
|
||||
usbutils
|
||||
|
||||
##
|
||||
# Build libcec for HDMI-CEC
|
||||
ARG LIBCEC_VERSION=6.0.2
|
||||
RUN apk add --no-cache \
|
||||
eudev-libs \
|
||||
p8-platform \
|
||||
&& apk add --no-cache --virtual .build-dependencies \
|
||||
build-base \
|
||||
cmake \
|
||||
eudev-dev \
|
||||
swig \
|
||||
p8-platform-dev \
|
||||
linux-headers \
|
||||
&& git clone --depth 1 -b libcec-${LIBCEC_VERSION} https://github.com/Pulse-Eight/libcec /usr/src/libcec \
|
||||
&& cd /usr/src/libcec \
|
||||
&& mkdir -p /usr/src/libcec/build \
|
||||
&& cd /usr/src/libcec/build \
|
||||
&& cmake -DCMAKE_INSTALL_PREFIX:PATH=/usr/local \
|
||||
-DPYTHON_LIBRARY="/usr/local/lib/libpython3.8.so" \
|
||||
-DPYTHON_INCLUDE_DIR="/usr/local/include/python3.8" \
|
||||
-DHAVE_LINUX_API=1 \
|
||||
.. \
|
||||
&& make -j$(nproc) \
|
||||
&& make install \
|
||||
&& echo "cec" > "/usr/local/lib/python3.8/site-packages/cec.pth" \
|
||||
&& apk del .build-dependencies \
|
||||
&& rm -rf /usr/src/libcec*
|
33
machine/qemux86-64
Normal file
33
machine/qemux86-64
Normal file
@@ -0,0 +1,33 @@
|
||||
ARG BUILD_VERSION
|
||||
FROM homeassistant/amd64-homeassistant:$BUILD_VERSION
|
||||
|
||||
RUN apk --no-cache add \
|
||||
usbutils
|
||||
|
||||
##
|
||||
# Build libcec for HDMI-CEC
|
||||
ARG LIBCEC_VERSION=6.0.2
|
||||
RUN apk add --no-cache \
|
||||
eudev-libs \
|
||||
p8-platform \
|
||||
&& apk add --no-cache --virtual .build-dependencies \
|
||||
build-base \
|
||||
cmake \
|
||||
eudev-dev \
|
||||
swig \
|
||||
p8-platform-dev \
|
||||
linux-headers \
|
||||
&& git clone --depth 1 -b libcec-${LIBCEC_VERSION} https://github.com/Pulse-Eight/libcec /usr/src/libcec \
|
||||
&& cd /usr/src/libcec \
|
||||
&& mkdir -p /usr/src/libcec/build \
|
||||
&& cd /usr/src/libcec/build \
|
||||
&& cmake -DCMAKE_INSTALL_PREFIX:PATH=/usr/local \
|
||||
-DPYTHON_LIBRARY="/usr/local/lib/libpython3.8.so" \
|
||||
-DPYTHON_INCLUDE_DIR="/usr/local/include/python3.8" \
|
||||
-DHAVE_LINUX_API=1 \
|
||||
.. \
|
||||
&& make -j$(nproc) \
|
||||
&& make install \
|
||||
&& echo "cec" > "/usr/local/lib/python3.8/site-packages/cec.pth" \
|
||||
&& apk del .build-dependencies \
|
||||
&& rm -rf /usr/src/libcec*
|
61
machine/raspberrypi
Normal file
61
machine/raspberrypi
Normal file
@@ -0,0 +1,61 @@
|
||||
ARG BUILD_VERSION
|
||||
FROM homeassistant/armhf-homeassistant:$BUILD_VERSION
|
||||
|
||||
RUN apk --no-cache add \
|
||||
raspberrypi \
|
||||
raspberrypi-libs \
|
||||
usbutils \
|
||||
&& sed -i "s|# RPi.GPIO|RPi.GPIO|g" /usr/src/homeassistant/requirements_all.txt \
|
||||
&& pip3 install --no-cache-dir --no-index --only-binary=:all: --find-links "${WHEELS_LINKS}" \
|
||||
RPi.GPIO -c /usr/src/homeassistant/requirements_all.txt
|
||||
|
||||
##
|
||||
# Set symlinks for raspberry pi camera binaries.
|
||||
RUN ln -sv /opt/vc/bin/raspistill /usr/local/bin/raspistill \
|
||||
&& ln -sv /opt/vc/bin/raspivid /usr/local/bin/raspivid \
|
||||
&& ln -sv /opt/vc/bin/raspividyuv /usr/local/bin/raspividyuv \
|
||||
&& ln -sv /opt/vc/bin/raspiyuv /usr/local/bin/raspiyuv
|
||||
|
||||
##
|
||||
# Build libcec with RPi support for HDMI-CEC
|
||||
ARG LIBCEC_VERSION=6.0.2
|
||||
RUN apk add --no-cache \
|
||||
eudev-libs \
|
||||
p8-platform \
|
||||
&& apk add --no-cache --virtual .build-dependencies \
|
||||
build-base \
|
||||
cmake \
|
||||
eudev-dev \
|
||||
swig \
|
||||
raspberrypi-dev \
|
||||
p8-platform-dev \
|
||||
&& git clone --depth 1 -b libcec-${LIBCEC_VERSION} https://github.com/Pulse-Eight/libcec /usr/src/libcec \
|
||||
&& mkdir -p /usr/src/libcec/build \
|
||||
&& cd /usr/src/libcec/build \
|
||||
&& cmake -DCMAKE_INSTALL_PREFIX:PATH=/usr/local \
|
||||
-DRPI_INCLUDE_DIR=/opt/vc/include \
|
||||
-DRPI_LIB_DIR=/opt/vc/lib \
|
||||
-DPYTHON_LIBRARY="/usr/local/lib/libpython3.8.so" \
|
||||
-DPYTHON_INCLUDE_DIR="/usr/local/include/python3.8" \
|
||||
.. \
|
||||
&& make -j$(nproc) \
|
||||
&& make install \
|
||||
&& echo "cec" > "/usr/local/lib/python3.8/site-packages/cec.pth" \
|
||||
&& apk del .build-dependencies \
|
||||
&& rm -rf /usr/src/libcec
|
||||
ENV LD_LIBRARY_PATH=/opt/vc/lib:${LD_LIBRARY_PATH}
|
||||
|
||||
##
|
||||
# Install DHT
|
||||
RUN apk add --no-cache --virtual .build-dependencies \
|
||||
gcc libc-dev raspberrypi-dev \
|
||||
&& export DHT_VERSION="$(cat /usr/src/homeassistant/requirements_all.txt | sed -n 's|.*Adafruit-DHT==\([0-9\.]*\).*|\1|p')" \
|
||||
&& git clone --depth 1 -b ${DHT_VERSION} https://github.com/adafruit/Adafruit_Python_DHT /usr/src/dht \
|
||||
&& cd /usr/src/dht \
|
||||
&& sed -i 's/^pi_version\ =\ None/pi_version\ =\ 1/' setup.py \
|
||||
&& sed -i 's/^platform\ =\ platform_detect.UNKNOWN/platform\ =\ platform_detect.RASPBERRY_PI/' setup.py \
|
||||
&& sed -i 's/platform\ =\ platform_detect.platform_detect()/pass/' setup.py \
|
||||
&& export MAKEFLAGS="-j$(nproc)" \
|
||||
&& pip3 install . \
|
||||
&& apk del .build-dependencies \
|
||||
&& rm -rf /usr/src/dht
|
61
machine/raspberrypi2
Normal file
61
machine/raspberrypi2
Normal file
@@ -0,0 +1,61 @@
|
||||
ARG BUILD_VERSION
|
||||
FROM homeassistant/armv7-homeassistant:$BUILD_VERSION
|
||||
|
||||
RUN apk --no-cache add \
|
||||
raspberrypi \
|
||||
raspberrypi-libs \
|
||||
usbutils \
|
||||
&& sed -i "s|# RPi.GPIO|RPi.GPIO|g" /usr/src/homeassistant/requirements_all.txt \
|
||||
&& pip3 install --no-cache-dir --no-index --only-binary=:all: --find-links "${WHEELS_LINKS}" \
|
||||
RPi.GPIO -c /usr/src/homeassistant/requirements_all.txt
|
||||
|
||||
##
|
||||
# Set symlinks for raspberry pi binaries.
|
||||
RUN ln -sv /opt/vc/bin/raspistill /usr/local/bin/raspistill \
|
||||
&& ln -sv /opt/vc/bin/raspivid /usr/local/bin/raspivid \
|
||||
&& ln -sv /opt/vc/bin/raspividyuv /usr/local/bin/raspividyuv \
|
||||
&& ln -sv /opt/vc/bin/raspiyuv /usr/local/bin/raspiyuv
|
||||
|
||||
##
|
||||
# Build libcec with RPi support for HDMI-CEC
|
||||
ARG LIBCEC_VERSION=6.0.2
|
||||
RUN apk add --no-cache \
|
||||
eudev-libs \
|
||||
p8-platform \
|
||||
&& apk add --no-cache --virtual .build-dependencies \
|
||||
build-base \
|
||||
cmake \
|
||||
eudev-dev \
|
||||
swig \
|
||||
raspberrypi-dev \
|
||||
p8-platform-dev \
|
||||
&& git clone --depth 1 -b libcec-${LIBCEC_VERSION} https://github.com/Pulse-Eight/libcec /usr/src/libcec \
|
||||
&& mkdir -p /usr/src/libcec/build \
|
||||
&& cd /usr/src/libcec/build \
|
||||
&& cmake -DCMAKE_INSTALL_PREFIX:PATH=/usr/local \
|
||||
-DRPI_INCLUDE_DIR=/opt/vc/include \
|
||||
-DRPI_LIB_DIR=/opt/vc/lib \
|
||||
-DPYTHON_LIBRARY="/usr/local/lib/libpython3.8.so" \
|
||||
-DPYTHON_INCLUDE_DIR="/usr/local/include/python3.8" \
|
||||
.. \
|
||||
&& make -j$(nproc) \
|
||||
&& make install \
|
||||
&& echo "cec" > "/usr/local/lib/python3.8/site-packages/cec.pth" \
|
||||
&& apk del .build-dependencies \
|
||||
&& rm -rf /usr/src/libcec
|
||||
ENV LD_LIBRARY_PATH=/opt/vc/lib:${LD_LIBRARY_PATH}
|
||||
|
||||
##
|
||||
# Install DHT
|
||||
RUN apk add --no-cache --virtual .build-dependencies \
|
||||
gcc libc-dev raspberrypi-dev \
|
||||
&& export DHT_VERSION="$(cat /usr/src/homeassistant/requirements_all.txt | sed -n 's|.*Adafruit-DHT==\([0-9\.]*\).*|\1|p')" \
|
||||
&& git clone --depth 1 -b ${DHT_VERSION} https://github.com/adafruit/Adafruit_Python_DHT /usr/src/dht \
|
||||
&& cd /usr/src/dht \
|
||||
&& sed -i 's/^pi_version\ =\ None/pi_version\ =\ 2/' setup.py \
|
||||
&& sed -i 's/^platform\ =\ platform_detect.UNKNOWN/platform\ =\ platform_detect.RASPBERRY_PI/' setup.py \
|
||||
&& sed -i 's/platform\ =\ platform_detect.platform_detect()/pass/' setup.py \
|
||||
&& export MAKEFLAGS="-j$(nproc)" \
|
||||
&& pip3 install . \
|
||||
&& apk del .build-dependencies \
|
||||
&& rm -rf /usr/src/dht
|
61
machine/raspberrypi3
Normal file
61
machine/raspberrypi3
Normal file
@@ -0,0 +1,61 @@
|
||||
ARG BUILD_VERSION
|
||||
FROM homeassistant/armv7-homeassistant:$BUILD_VERSION
|
||||
|
||||
RUN apk --no-cache add \
|
||||
raspberrypi \
|
||||
raspberrypi-libs \
|
||||
usbutils \
|
||||
&& sed -i "s|# RPi.GPIO|RPi.GPIO|g" /usr/src/homeassistant/requirements_all.txt \
|
||||
&& pip3 install --no-cache-dir --no-index --only-binary=:all: --find-links "${WHEELS_LINKS}" \
|
||||
RPi.GPIO bluepy pybluez -c /usr/src/homeassistant/requirements_all.txt
|
||||
|
||||
##
|
||||
# Set symlinks for raspberry pi binaries.
|
||||
RUN ln -sv /opt/vc/bin/raspistill /usr/local/bin/raspistill \
|
||||
&& ln -sv /opt/vc/bin/raspivid /usr/local/bin/raspivid \
|
||||
&& ln -sv /opt/vc/bin/raspividyuv /usr/local/bin/raspividyuv \
|
||||
&& ln -sv /opt/vc/bin/raspiyuv /usr/local/bin/raspiyuv
|
||||
|
||||
##
|
||||
# Build libcec with RPi support for HDMI-CEC
|
||||
ARG LIBCEC_VERSION=6.0.2
|
||||
RUN apk add --no-cache \
|
||||
eudev-libs \
|
||||
p8-platform \
|
||||
&& apk add --no-cache --virtual .build-dependencies \
|
||||
build-base \
|
||||
cmake \
|
||||
eudev-dev \
|
||||
swig \
|
||||
raspberrypi-dev \
|
||||
p8-platform-dev \
|
||||
&& git clone --depth 1 -b libcec-${LIBCEC_VERSION} https://github.com/Pulse-Eight/libcec /usr/src/libcec \
|
||||
&& mkdir -p /usr/src/libcec/build \
|
||||
&& cd /usr/src/libcec/build \
|
||||
&& cmake -DCMAKE_INSTALL_PREFIX:PATH=/usr/local \
|
||||
-DRPI_INCLUDE_DIR=/opt/vc/include \
|
||||
-DRPI_LIB_DIR=/opt/vc/lib \
|
||||
-DPYTHON_LIBRARY="/usr/local/lib/libpython3.8.so" \
|
||||
-DPYTHON_INCLUDE_DIR="/usr/local/include/python3.8" \
|
||||
.. \
|
||||
&& make -j$(nproc) \
|
||||
&& make install \
|
||||
&& echo "cec" > "/usr/local/lib/python3.8/site-packages/cec.pth" \
|
||||
&& apk del .build-dependencies \
|
||||
&& rm -rf /usr/src/libcec
|
||||
ENV LD_LIBRARY_PATH=/opt/vc/lib:${LD_LIBRARY_PATH}
|
||||
|
||||
##
|
||||
# Install DHT
|
||||
RUN apk add --no-cache --virtual .build-dependencies \
|
||||
gcc libc-dev raspberrypi-dev \
|
||||
&& export DHT_VERSION="$(cat /usr/src/homeassistant/requirements_all.txt | sed -n 's|.*Adafruit-DHT==\([0-9\.]*\).*|\1|p')" \
|
||||
&& git clone --depth 1 -b ${DHT_VERSION} https://github.com/adafruit/Adafruit_Python_DHT /usr/src/dht \
|
||||
&& cd /usr/src/dht \
|
||||
&& sed -i 's/^pi_version\ =\ None/pi_version\ =\ 3/' setup.py \
|
||||
&& sed -i 's/^platform\ =\ platform_detect.UNKNOWN/platform\ =\ platform_detect.RASPBERRY_PI/' setup.py \
|
||||
&& sed -i 's/platform\ =\ platform_detect.platform_detect()/pass/' setup.py \
|
||||
&& export MAKEFLAGS="-j$(nproc)" \
|
||||
&& pip3 install . \
|
||||
&& apk del .build-dependencies \
|
||||
&& rm -rf /usr/src/dht
|
61
machine/raspberrypi3-64
Normal file
61
machine/raspberrypi3-64
Normal file
@@ -0,0 +1,61 @@
|
||||
ARG BUILD_VERSION
|
||||
FROM homeassistant/aarch64-homeassistant:$BUILD_VERSION
|
||||
|
||||
RUN apk --no-cache add \
|
||||
raspberrypi \
|
||||
raspberrypi-libs \
|
||||
usbutils \
|
||||
&& sed -i "s|# RPi.GPIO|RPi.GPIO|g" /usr/src/homeassistant/requirements_all.txt \
|
||||
&& pip3 install --no-cache-dir --no-index --only-binary=:all: --find-links "${WHEELS_LINKS}" \
|
||||
RPi.GPIO bluepy pybluez -c /usr/src/homeassistant/requirements_all.txt
|
||||
|
||||
##
|
||||
# Set symlinks for raspberry pi binaries.
|
||||
RUN ln -sv /opt/vc/bin/raspistill /usr/local/bin/raspistill \
|
||||
&& ln -sv /opt/vc/bin/raspivid /usr/local/bin/raspivid \
|
||||
&& ln -sv /opt/vc/bin/raspividyuv /usr/local/bin/raspividyuv \
|
||||
&& ln -sv /opt/vc/bin/raspiyuv /usr/local/bin/raspiyuv
|
||||
|
||||
##
|
||||
# Build libcec with RPi support for HDMI-CEC
|
||||
ARG LIBCEC_VERSION=6.0.2
|
||||
RUN apk add --no-cache \
|
||||
eudev-libs \
|
||||
p8-platform \
|
||||
&& apk add --no-cache --virtual .build-dependencies \
|
||||
build-base \
|
||||
cmake \
|
||||
eudev-dev \
|
||||
swig \
|
||||
raspberrypi-dev \
|
||||
p8-platform-dev \
|
||||
&& git clone --depth 1 -b libcec-${LIBCEC_VERSION} https://github.com/Pulse-Eight/libcec /usr/src/libcec \
|
||||
&& mkdir -p /usr/src/libcec/build \
|
||||
&& cd /usr/src/libcec/build \
|
||||
&& cmake -DCMAKE_INSTALL_PREFIX:PATH=/usr/local \
|
||||
-DRPI_INCLUDE_DIR=/opt/vc/include \
|
||||
-DRPI_LIB_DIR=/opt/vc/lib \
|
||||
-DPYTHON_LIBRARY="/usr/local/lib/libpython3.8.so" \
|
||||
-DPYTHON_INCLUDE_DIR="/usr/local/include/python3.8" \
|
||||
.. \
|
||||
&& make -j$(nproc) \
|
||||
&& make install \
|
||||
&& echo "cec" > "/usr/local/lib/python3.8/site-packages/cec.pth" \
|
||||
&& apk del .build-dependencies \
|
||||
&& rm -rf /usr/src/libcec
|
||||
ENV LD_LIBRARY_PATH=/opt/vc/lib:${LD_LIBRARY_PATH}
|
||||
|
||||
##
|
||||
# Install DHT
|
||||
RUN apk add --no-cache --virtual .build-dependencies \
|
||||
gcc libc-dev raspberrypi-dev \
|
||||
&& export DHT_VERSION="$(cat /usr/src/homeassistant/requirements_all.txt | sed -n 's|.*Adafruit-DHT==\([0-9\.]*\).*|\1|p')" \
|
||||
&& git clone --depth 1 -b ${DHT_VERSION} https://github.com/adafruit/Adafruit_Python_DHT /usr/src/dht \
|
||||
&& cd /usr/src/dht \
|
||||
&& sed -i 's/^pi_version\ =\ None/pi_version\ =\ 3/' setup.py \
|
||||
&& sed -i 's/^platform\ =\ platform_detect.UNKNOWN/platform\ =\ platform_detect.RASPBERRY_PI/' setup.py \
|
||||
&& sed -i 's/platform\ =\ platform_detect.platform_detect()/pass/' setup.py \
|
||||
&& export MAKEFLAGS="-j$(nproc)" \
|
||||
&& pip3 install . \
|
||||
&& apk del .build-dependencies \
|
||||
&& rm -rf /usr/src/dht
|
61
machine/raspberrypi4
Normal file
61
machine/raspberrypi4
Normal file
@@ -0,0 +1,61 @@
|
||||
ARG BUILD_VERSION
|
||||
FROM homeassistant/armv7-homeassistant:$BUILD_VERSION
|
||||
|
||||
RUN apk --no-cache add \
|
||||
raspberrypi \
|
||||
raspberrypi-libs \
|
||||
usbutils \
|
||||
&& sed -i "s|# RPi.GPIO|RPi.GPIO|g" /usr/src/homeassistant/requirements_all.txt \
|
||||
&& pip3 install --no-cache-dir --no-index --only-binary=:all: --find-links "${WHEELS_LINKS}" \
|
||||
RPi.GPIO bluepy pybluez -c /usr/src/homeassistant/requirements_all.txt
|
||||
|
||||
##
|
||||
# Set symlinks for raspberry pi binaries.
|
||||
RUN ln -sv /opt/vc/bin/raspistill /usr/local/bin/raspistill \
|
||||
&& ln -sv /opt/vc/bin/raspivid /usr/local/bin/raspivid \
|
||||
&& ln -sv /opt/vc/bin/raspividyuv /usr/local/bin/raspividyuv \
|
||||
&& ln -sv /opt/vc/bin/raspiyuv /usr/local/bin/raspiyuv
|
||||
|
||||
##
|
||||
# Build libcec with RPi support for HDMI-CEC
|
||||
ARG LIBCEC_VERSION=6.0.2
|
||||
RUN apk add --no-cache \
|
||||
eudev-libs \
|
||||
p8-platform \
|
||||
&& apk add --no-cache --virtual .build-dependencies \
|
||||
build-base \
|
||||
cmake \
|
||||
eudev-dev \
|
||||
swig \
|
||||
raspberrypi-dev \
|
||||
p8-platform-dev \
|
||||
&& git clone --depth 1 -b libcec-${LIBCEC_VERSION} https://github.com/Pulse-Eight/libcec /usr/src/libcec \
|
||||
&& mkdir -p /usr/src/libcec/build \
|
||||
&& cd /usr/src/libcec/build \
|
||||
&& cmake -DCMAKE_INSTALL_PREFIX:PATH=/usr/local \
|
||||
-DRPI_INCLUDE_DIR=/opt/vc/include \
|
||||
-DRPI_LIB_DIR=/opt/vc/lib \
|
||||
-DPYTHON_LIBRARY="/usr/local/lib/libpython3.8.so" \
|
||||
-DPYTHON_INCLUDE_DIR="/usr/local/include/python3.8" \
|
||||
.. \
|
||||
&& make -j$(nproc) \
|
||||
&& make install \
|
||||
&& echo "cec" > "/usr/local/lib/python3.8/site-packages/cec.pth" \
|
||||
&& apk del .build-dependencies \
|
||||
&& rm -rf /usr/src/libcec
|
||||
ENV LD_LIBRARY_PATH=/opt/vc/lib:${LD_LIBRARY_PATH}
|
||||
|
||||
##
|
||||
# Install DHT
|
||||
RUN apk add --no-cache --virtual .build-dependencies \
|
||||
gcc libc-dev raspberrypi-dev \
|
||||
&& export DHT_VERSION="$(cat /usr/src/homeassistant/requirements_all.txt | sed -n 's|.*Adafruit-DHT==\([0-9\.]*\).*|\1|p')" \
|
||||
&& git clone --depth 1 -b ${DHT_VERSION} https://github.com/adafruit/Adafruit_Python_DHT /usr/src/dht \
|
||||
&& cd /usr/src/dht \
|
||||
&& sed -i 's/^pi_version\ =\ None/pi_version\ =\ 3/' setup.py \
|
||||
&& sed -i 's/^platform\ =\ platform_detect.UNKNOWN/platform\ =\ platform_detect.RASPBERRY_PI/' setup.py \
|
||||
&& sed -i 's/platform\ =\ platform_detect.platform_detect()/pass/' setup.py \
|
||||
&& export MAKEFLAGS="-j$(nproc)" \
|
||||
&& pip3 install . \
|
||||
&& apk del .build-dependencies \
|
||||
&& rm -rf /usr/src/dht
|
61
machine/raspberrypi4-64
Normal file
61
machine/raspberrypi4-64
Normal file
@@ -0,0 +1,61 @@
|
||||
ARG BUILD_VERSION
|
||||
FROM homeassistant/aarch64-homeassistant:$BUILD_VERSION
|
||||
|
||||
RUN apk --no-cache add \
|
||||
raspberrypi \
|
||||
raspberrypi-libs \
|
||||
usbutils \
|
||||
&& sed -i "s|# RPi.GPIO|RPi.GPIO|g" /usr/src/homeassistant/requirements_all.txt \
|
||||
&& pip3 install --no-cache-dir --no-index --only-binary=:all: --find-links "${WHEELS_LINKS}" \
|
||||
RPi.GPIO bluepy pybluez -c /usr/src/homeassistant/requirements_all.txt
|
||||
|
||||
##
|
||||
# Set symlinks for raspberry pi binaries.
|
||||
RUN ln -sv /opt/vc/bin/raspistill /usr/local/bin/raspistill \
|
||||
&& ln -sv /opt/vc/bin/raspivid /usr/local/bin/raspivid \
|
||||
&& ln -sv /opt/vc/bin/raspividyuv /usr/local/bin/raspividyuv \
|
||||
&& ln -sv /opt/vc/bin/raspiyuv /usr/local/bin/raspiyuv
|
||||
|
||||
##
|
||||
# Build libcec with RPi support for HDMI-CEC
|
||||
ARG LIBCEC_VERSION=6.0.2
|
||||
RUN apk add --no-cache \
|
||||
eudev-libs \
|
||||
p8-platform \
|
||||
&& apk add --no-cache --virtual .build-dependencies \
|
||||
build-base \
|
||||
cmake \
|
||||
eudev-dev \
|
||||
swig \
|
||||
raspberrypi-dev \
|
||||
p8-platform-dev \
|
||||
&& git clone --depth 1 -b libcec-${LIBCEC_VERSION} https://github.com/Pulse-Eight/libcec /usr/src/libcec \
|
||||
&& mkdir -p /usr/src/libcec/build \
|
||||
&& cd /usr/src/libcec/build \
|
||||
&& cmake -DCMAKE_INSTALL_PREFIX:PATH=/usr/local \
|
||||
-DRPI_INCLUDE_DIR=/opt/vc/include \
|
||||
-DRPI_LIB_DIR=/opt/vc/lib \
|
||||
-DPYTHON_LIBRARY="/usr/local/lib/libpython3.8.so" \
|
||||
-DPYTHON_INCLUDE_DIR="/usr/local/include/python3.8" \
|
||||
.. \
|
||||
&& make -j$(nproc) \
|
||||
&& make install \
|
||||
&& echo "cec" > "/usr/local/lib/python3.8/site-packages/cec.pth" \
|
||||
&& apk del .build-dependencies \
|
||||
&& rm -rf /usr/src/libcec
|
||||
ENV LD_LIBRARY_PATH=/opt/vc/lib:${LD_LIBRARY_PATH}
|
||||
|
||||
##
|
||||
# Install DHT
|
||||
RUN apk add --no-cache --virtual .build-dependencies \
|
||||
gcc libc-dev raspberrypi-dev \
|
||||
&& export DHT_VERSION="$(cat /usr/src/homeassistant/requirements_all.txt | sed -n 's|.*Adafruit-DHT==\([0-9\.]*\).*|\1|p')" \
|
||||
&& git clone --depth 1 -b ${DHT_VERSION} https://github.com/adafruit/Adafruit_Python_DHT /usr/src/dht \
|
||||
&& cd /usr/src/dht \
|
||||
&& sed -i 's/^pi_version\ =\ None/pi_version\ =\ 3/' setup.py \
|
||||
&& sed -i 's/^platform\ =\ platform_detect.UNKNOWN/platform\ =\ platform_detect.RASPBERRY_PI/' setup.py \
|
||||
&& sed -i 's/platform\ =\ platform_detect.platform_detect()/pass/' setup.py \
|
||||
&& export MAKEFLAGS="-j$(nproc)" \
|
||||
&& pip3 install . \
|
||||
&& apk del .build-dependencies \
|
||||
&& rm -rf /usr/src/dht
|
48
machine/tinker
Normal file
48
machine/tinker
Normal file
@@ -0,0 +1,48 @@
|
||||
ARG BUILD_VERSION
|
||||
FROM homeassistant/armv7-homeassistant:$BUILD_VERSION
|
||||
|
||||
RUN apk --no-cache add usbutils \
|
||||
&& pip3 install --no-cache-dir --no-index --only-binary=:all: --find-links "${WHEELS_LINKS}" \
|
||||
-c /usr/src/homeassistant/homeassistant/package_constraints.txt \
|
||||
bluepy \
|
||||
pybluez \
|
||||
pygatt[GATTTOOL]
|
||||
|
||||
# Install GPIO support
|
||||
RUN apk add --no-cache --virtual .build-dependencies \
|
||||
gcc libc-dev musl-dev \
|
||||
&& git clone --depth 1 https://github.com/TinkerBoard/gpio_lib_python /usr/src/gpio \
|
||||
&& cd /usr/src/gpio \
|
||||
&& sed -i "s/caddr_t/void*/g" source/wiringTB.c \
|
||||
&& export MAKEFLAGS="-j$(nproc)" \
|
||||
&& python3 setup.py install \
|
||||
&& apk del .build-dependencies \
|
||||
&& rm -rf /usr/src/gpio
|
||||
|
||||
##
|
||||
# Build libcec for HDMI-CEC
|
||||
ARG LIBCEC_VERSION=6.0.2
|
||||
RUN apk add --no-cache \
|
||||
eudev-libs \
|
||||
p8-platform \
|
||||
&& apk add --no-cache --virtual .build-dependencies \
|
||||
build-base \
|
||||
cmake \
|
||||
eudev-dev \
|
||||
swig \
|
||||
p8-platform-dev \
|
||||
linux-headers \
|
||||
&& git clone --depth 1 -b libcec-${LIBCEC_VERSION} https://github.com/Pulse-Eight/libcec /usr/src/libcec \
|
||||
&& cd /usr/src/libcec \
|
||||
&& mkdir -p /usr/src/libcec/build \
|
||||
&& cd /usr/src/libcec/build \
|
||||
&& cmake -DCMAKE_INSTALL_PREFIX:PATH=/usr/local \
|
||||
-DPYTHON_LIBRARY="/usr/local/lib/libpython3.8.so" \
|
||||
-DPYTHON_INCLUDE_DIR="/usr/local/include/python3.8" \
|
||||
-DHAVE_LINUX_API=1 \
|
||||
.. \
|
||||
&& make -j$(nproc) \
|
||||
&& make install \
|
||||
&& echo "cec" > "/usr/local/lib/python3.8/site-packages/cec.pth" \
|
||||
&& apk del .build-dependencies \
|
||||
&& rm -rf /usr/src/libcec*
|
@@ -194,7 +194,7 @@ aioimaplib==0.7.15
|
||||
aiokafka==0.6.0
|
||||
|
||||
# homeassistant.components.kef
|
||||
aiokef==0.2.13
|
||||
aiokef==0.2.16
|
||||
|
||||
# homeassistant.components.lifx
|
||||
aiolifx==0.6.7
|
||||
@@ -275,7 +275,7 @@ aqualogic==1.0
|
||||
arcam-fmj==0.5.3
|
||||
|
||||
# homeassistant.components.arris_tg2492lg
|
||||
arris-tg2492lg==1.0.0
|
||||
arris-tg2492lg==1.1.0
|
||||
|
||||
# homeassistant.components.ampio
|
||||
asmog==0.0.6
|
||||
@@ -753,7 +753,7 @@ hole==0.5.1
|
||||
holidays==0.10.3
|
||||
|
||||
# homeassistant.components.frontend
|
||||
home-assistant-frontend==20201001.1
|
||||
home-assistant-frontend==20201001.2
|
||||
|
||||
# homeassistant.components.zwave
|
||||
homeassistant-pyozw==0.1.10
|
||||
@@ -1058,6 +1058,8 @@ orvibo==1.1.1
|
||||
ovoenergy==1.1.7
|
||||
|
||||
# homeassistant.components.mqtt
|
||||
paho-mqtt==1.5.0
|
||||
|
||||
# homeassistant.components.shiftr
|
||||
paho-mqtt==1.5.1
|
||||
|
||||
@@ -1250,7 +1252,7 @@ pyarlo==0.2.3
|
||||
pyatag==0.3.4.4
|
||||
|
||||
# homeassistant.components.netatmo
|
||||
pyatmo==4.0.0
|
||||
pyatmo==4.1.0
|
||||
|
||||
# homeassistant.components.atome
|
||||
pyatome==0.1.1
|
||||
@@ -1401,7 +1403,7 @@ pyheos==0.6.0
|
||||
pyhik==0.2.7
|
||||
|
||||
# homeassistant.components.hive
|
||||
pyhiveapi==0.2.20.1
|
||||
pyhiveapi==0.2.20.2
|
||||
|
||||
# homeassistant.components.homematic
|
||||
pyhomematic==0.1.68
|
||||
@@ -2293,7 +2295,7 @@ xs1-api-client==3.0.0
|
||||
yalesmartalarmclient==0.1.6
|
||||
|
||||
# homeassistant.components.yeelight
|
||||
yeelight==0.5.3
|
||||
yeelight==0.5.4
|
||||
|
||||
# homeassistant.components.yeelightsunflower
|
||||
yeelightsunflower==0.0.10
|
||||
|
@@ -376,7 +376,7 @@ hole==0.5.1
|
||||
holidays==0.10.3
|
||||
|
||||
# homeassistant.components.frontend
|
||||
home-assistant-frontend==20201001.1
|
||||
home-assistant-frontend==20201001.2
|
||||
|
||||
# homeassistant.components.zwave
|
||||
homeassistant-pyozw==0.1.10
|
||||
@@ -499,8 +499,7 @@ openerz-api==0.1.0
|
||||
ovoenergy==1.1.7
|
||||
|
||||
# homeassistant.components.mqtt
|
||||
# homeassistant.components.shiftr
|
||||
paho-mqtt==1.5.1
|
||||
paho-mqtt==1.5.0
|
||||
|
||||
# homeassistant.components.panasonic_viera
|
||||
panasonic_viera==0.3.6
|
||||
@@ -616,7 +615,7 @@ pyarlo==0.2.3
|
||||
pyatag==0.3.4.4
|
||||
|
||||
# homeassistant.components.netatmo
|
||||
pyatmo==4.0.0
|
||||
pyatmo==4.1.0
|
||||
|
||||
# homeassistant.components.blackbird
|
||||
pyblackbird==0.5
|
||||
@@ -1065,7 +1064,7 @@ wolf_smartset==0.1.6
|
||||
xmltodict==0.12.0
|
||||
|
||||
# homeassistant.components.yeelight
|
||||
yeelight==0.5.3
|
||||
yeelight==0.5.4
|
||||
|
||||
# homeassistant.components.zeroconf
|
||||
zeroconf==0.28.5
|
||||
|
@@ -1213,3 +1213,66 @@ async def test_group_that_references_two_types_of_groups(hass):
|
||||
assert hass.states.get("group.covers").state == "closed"
|
||||
assert hass.states.get("group.device_trackers").state == "home"
|
||||
assert hass.states.get("group.grouped_group").state == "on"
|
||||
|
||||
|
||||
async def test_plant_group(hass):
|
||||
"""Test plant states can be grouped."""
|
||||
|
||||
entity_ids = [
|
||||
"plant.upstairs",
|
||||
"plant.downstairs",
|
||||
]
|
||||
|
||||
assert await async_setup_component(
|
||||
hass,
|
||||
"plant",
|
||||
{
|
||||
"plant": {
|
||||
"plantname": {
|
||||
"sensors": {
|
||||
"moisture": "sensor.mqtt_plant_moisture",
|
||||
"battery": "sensor.mqtt_plant_battery",
|
||||
"temperature": "sensor.mqtt_plant_temperature",
|
||||
"conductivity": "sensor.mqtt_plant_conductivity",
|
||||
"brightness": "sensor.mqtt_plant_brightness",
|
||||
},
|
||||
"min_moisture": 20,
|
||||
"max_moisture": 60,
|
||||
"min_battery": 17,
|
||||
"min_conductivity": 500,
|
||||
"min_temperature": 15,
|
||||
"min_brightness": 500,
|
||||
}
|
||||
}
|
||||
},
|
||||
)
|
||||
assert await async_setup_component(
|
||||
hass,
|
||||
"group",
|
||||
{
|
||||
"group": {
|
||||
"plants": {"entities": entity_ids},
|
||||
"plant_with_binary_sensors": {
|
||||
"entities": [*entity_ids, "binary_sensor.planter"]
|
||||
},
|
||||
}
|
||||
},
|
||||
)
|
||||
await hass.async_block_till_done()
|
||||
|
||||
hass.states.async_set("binary_sensor.planter", "off")
|
||||
for entity_id in entity_ids:
|
||||
hass.states.async_set(entity_id, "ok")
|
||||
await hass.async_block_till_done()
|
||||
await hass.async_block_till_done()
|
||||
|
||||
assert hass.states.get("group.plants").state == "ok"
|
||||
assert hass.states.get("group.plant_with_binary_sensors").state == "off"
|
||||
|
||||
hass.states.async_set("binary_sensor.planter", "on")
|
||||
for entity_id in entity_ids:
|
||||
hass.states.async_set(entity_id, "problem")
|
||||
|
||||
await hass.async_block_till_done()
|
||||
assert hass.states.get("group.plants").state == "problem"
|
||||
assert hass.states.get("group.plant_with_binary_sensors").state == "on"
|
||||
|
@@ -4,6 +4,7 @@ from datetime import datetime, timedelta
|
||||
import unittest
|
||||
|
||||
import pytest
|
||||
from sqlalchemy.exc import OperationalError
|
||||
|
||||
from homeassistant.components.recorder import (
|
||||
CONFIG_SCHEMA,
|
||||
@@ -452,3 +453,41 @@ def test_run_information(hass_recorder):
|
||||
|
||||
class CannotSerializeMe:
|
||||
"""A class that the JSONEncoder cannot serialize."""
|
||||
|
||||
|
||||
def test_saving_state_with_exception(hass, hass_recorder, caplog):
|
||||
"""Test saving and restoring a state."""
|
||||
hass = hass_recorder()
|
||||
|
||||
entity_id = "test.recorder"
|
||||
state = "restoring_from_db"
|
||||
attributes = {"test_attr": 5, "test_attr_10": "nice"}
|
||||
|
||||
def _throw_if_state_in_session(*args, **kwargs):
|
||||
for obj in hass.data[DATA_INSTANCE].event_session:
|
||||
if isinstance(obj, States):
|
||||
raise OperationalError(
|
||||
"insert the state", "fake params", "forced to fail"
|
||||
)
|
||||
|
||||
with patch("time.sleep"), patch.object(
|
||||
hass.data[DATA_INSTANCE].event_session,
|
||||
"flush",
|
||||
side_effect=_throw_if_state_in_session,
|
||||
):
|
||||
hass.states.set(entity_id, "fail", attributes)
|
||||
wait_recording_done(hass)
|
||||
|
||||
assert "Error executing query" in caplog.text
|
||||
assert "Error saving events" not in caplog.text
|
||||
|
||||
caplog.clear()
|
||||
hass.states.set(entity_id, state, attributes)
|
||||
wait_recording_done(hass)
|
||||
|
||||
with session_scope(hass=hass) as session:
|
||||
db_states = list(session.query(States))
|
||||
assert len(db_states) >= 1
|
||||
|
||||
assert "Error executing query" not in caplog.text
|
||||
assert "Error saving events" not in caplog.text
|
||||
|
@@ -488,6 +488,41 @@ async def test_render_template_with_error(hass, websocket_client, caplog):
|
||||
assert "TemplateError" not in caplog.text
|
||||
|
||||
|
||||
async def test_render_template_with_timeout_and_error(hass, websocket_client, caplog):
|
||||
"""Test a template with an error with a timeout."""
|
||||
await websocket_client.send_json(
|
||||
{
|
||||
"id": 5,
|
||||
"type": "render_template",
|
||||
"template": "{{ now() | rando }}",
|
||||
"timeout": 5,
|
||||
}
|
||||
)
|
||||
|
||||
msg = await websocket_client.receive_json()
|
||||
assert msg["id"] == 5
|
||||
assert msg["type"] == const.TYPE_RESULT
|
||||
assert not msg["success"]
|
||||
assert msg["error"]["code"] == const.ERR_TEMPLATE_ERROR
|
||||
|
||||
assert "TemplateError" not in caplog.text
|
||||
|
||||
|
||||
async def test_render_template_error_in_template_code(hass, websocket_client, caplog):
|
||||
"""Test a template that will throw in template.py."""
|
||||
await websocket_client.send_json(
|
||||
{"id": 5, "type": "render_template", "template": "{{ now() | random }}"}
|
||||
)
|
||||
|
||||
msg = await websocket_client.receive_json()
|
||||
assert msg["id"] == 5
|
||||
assert msg["type"] == const.TYPE_RESULT
|
||||
assert not msg["success"]
|
||||
assert msg["error"]["code"] == const.ERR_TEMPLATE_ERROR
|
||||
|
||||
assert "TemplateError" not in caplog.text
|
||||
|
||||
|
||||
async def test_render_template_with_delayed_error(hass, websocket_client, caplog):
|
||||
"""Test a template with an error that only happens after a state change."""
|
||||
hass.states.async_set("sensor.test", "on")
|
||||
|
@@ -486,6 +486,19 @@ async def test_async_track_state_change_event(hass):
|
||||
unsub_throws()
|
||||
|
||||
|
||||
async def test_async_track_state_change_event_with_empty_list(hass):
|
||||
"""Test async_track_state_change_event passing an empty list of entities."""
|
||||
unsub_single = async_track_state_change_event(
|
||||
hass, [], ha.callback(lambda event: None)
|
||||
)
|
||||
unsub_single2 = async_track_state_change_event(
|
||||
hass, [], ha.callback(lambda event: None)
|
||||
)
|
||||
|
||||
unsub_single2()
|
||||
unsub_single()
|
||||
|
||||
|
||||
async def test_async_track_state_added_domain(hass):
|
||||
"""Test async_track_state_added_domain."""
|
||||
single_entity_id_tracker = []
|
||||
@@ -568,6 +581,32 @@ async def test_async_track_state_added_domain(hass):
|
||||
unsub_throws()
|
||||
|
||||
|
||||
async def test_async_track_state_added_domain_with_empty_list(hass):
|
||||
"""Test async_track_state_added_domain passing an empty list of domains."""
|
||||
unsub_single = async_track_state_added_domain(
|
||||
hass, [], ha.callback(lambda event: None)
|
||||
)
|
||||
unsub_single2 = async_track_state_added_domain(
|
||||
hass, [], ha.callback(lambda event: None)
|
||||
)
|
||||
|
||||
unsub_single2()
|
||||
unsub_single()
|
||||
|
||||
|
||||
async def test_async_track_state_removed_domain_with_empty_list(hass):
|
||||
"""Test async_track_state_removed_domain passing an empty list of domains."""
|
||||
unsub_single = async_track_state_removed_domain(
|
||||
hass, [], ha.callback(lambda event: None)
|
||||
)
|
||||
unsub_single2 = async_track_state_removed_domain(
|
||||
hass, [], ha.callback(lambda event: None)
|
||||
)
|
||||
|
||||
unsub_single2()
|
||||
unsub_single()
|
||||
|
||||
|
||||
async def test_async_track_state_removed_domain(hass):
|
||||
"""Test async_track_state_removed_domain."""
|
||||
single_entity_id_tracker = []
|
||||
@@ -2876,3 +2915,16 @@ async def test_async_track_entity_registry_updated_event_with_a_callback_that_th
|
||||
unsub2()
|
||||
|
||||
assert event_data[0] == {"action": "create", "entity_id": "switch.puppy_feeder"}
|
||||
|
||||
|
||||
async def test_async_track_entity_registry_updated_event_with_empty_list(hass):
|
||||
"""Test async_track_entity_registry_updated_event passing an empty list of entities."""
|
||||
unsub_single = hass.helpers.event.async_track_entity_registry_updated_event(
|
||||
[], ha.callback(lambda event: None)
|
||||
)
|
||||
unsub_single2 = hass.helpers.event.async_track_entity_registry_updated_event(
|
||||
[], ha.callback(lambda event: None)
|
||||
)
|
||||
|
||||
unsub_single2()
|
||||
unsub_single()
|
||||
|
@@ -2539,6 +2539,16 @@ async def test_lights(hass):
|
||||
assert f"sensor{i}" in info.result()
|
||||
|
||||
|
||||
async def test_template_errors(hass):
|
||||
"""Test template rendering wraps exceptions with TemplateError."""
|
||||
|
||||
with pytest.raises(TemplateError):
|
||||
template.Template("{{ now() | rando }}", hass).async_render()
|
||||
|
||||
with pytest.raises(TemplateError):
|
||||
template.Template("{{ now() | random }}", hass).async_render()
|
||||
|
||||
|
||||
async def test_state_attributes(hass):
|
||||
"""Test state attributes."""
|
||||
hass.states.async_set("sensor.test", "23")
|
||||
|
Reference in New Issue
Block a user