Home Assistant Unofficial Reference 2024.12.1
image.py
Go to the documentation of this file.
1 """Support for image which integrates with other components."""
2 
3 from __future__ import annotations
4 
5 import logging
6 from typing import Any
7 
8 import voluptuous as vol
9 
10 from homeassistant.components.image import DOMAIN as IMAGE_DOMAIN, ImageEntity
11 from homeassistant.config_entries import ConfigEntry
12 from homeassistant.const import (
13  CONF_DEVICE_ID,
14  CONF_NAME,
15  CONF_UNIQUE_ID,
16  CONF_URL,
17  CONF_VERIFY_SSL,
18 )
19 from homeassistant.core import HomeAssistant, callback
20 from homeassistant.exceptions import TemplateError
21 from homeassistant.helpers import config_validation as cv, selector
22 from homeassistant.helpers.device import async_device_info_to_link_from_device_id
23 from homeassistant.helpers.entity_platform import AddEntitiesCallback
24 from homeassistant.helpers.typing import ConfigType, DiscoveryInfoType
25 from homeassistant.util import dt as dt_util
26 
27 from . import TriggerUpdateCoordinator
28 from .const import CONF_PICTURE
29 from .template_entity import TemplateEntity, make_template_entity_common_schema
30 from .trigger_entity import TriggerEntity
31 
32 _LOGGER = logging.getLogger(__name__)
33 
34 DEFAULT_NAME = "Template Image"
35 
36 GET_IMAGE_TIMEOUT = 10
37 
38 IMAGE_SCHEMA = vol.Schema(
39  {
40  vol.Required(CONF_URL): cv.template,
41  vol.Optional(CONF_VERIFY_SSL, default=True): bool,
42  }
43 ).extend(make_template_entity_common_schema(DEFAULT_NAME).schema)
44 
45 
46 IMAGE_CONFIG_SCHEMA = vol.Schema(
47  {
48  vol.Optional(CONF_NAME): cv.template,
49  vol.Required(CONF_URL): cv.template,
50  vol.Optional(CONF_VERIFY_SSL, default=True): bool,
51  vol.Optional(CONF_DEVICE_ID): selector.DeviceSelector(),
52  }
53 )
54 
55 
57  hass: HomeAssistant, definitions: list[dict[str, Any]], unique_id_prefix: str | None
58 ) -> list[StateImageEntity]:
59  """Create the template image."""
60  entities = []
61  for definition in definitions:
62  unique_id = definition.get(CONF_UNIQUE_ID)
63  if unique_id and unique_id_prefix:
64  unique_id = f"{unique_id_prefix}-{unique_id}"
65  entities.append(StateImageEntity(hass, definition, unique_id))
66  return entities
67 
68 
70  hass: HomeAssistant,
71  config: ConfigType,
72  async_add_entities: AddEntitiesCallback,
73  discovery_info: DiscoveryInfoType | None = None,
74 ) -> None:
75  """Set up the template image."""
76  if discovery_info is None:
77  _LOGGER.warning(
78  "Template image entities can only be configured under template:"
79  )
80  return
81 
82  if "coordinator" in discovery_info:
84  TriggerImageEntity(hass, discovery_info["coordinator"], config)
85  for config in discovery_info["entities"]
86  )
87  return
88 
91  hass, discovery_info["entities"], discovery_info["unique_id"]
92  )
93  )
94 
95 
97  hass: HomeAssistant,
98  config_entry: ConfigEntry,
99  async_add_entities: AddEntitiesCallback,
100 ) -> None:
101  """Initialize config entry."""
102  _options = dict(config_entry.options)
103  _options.pop("template_type")
104  validated_config = IMAGE_CONFIG_SCHEMA(_options)
106  [StateImageEntity(hass, validated_config, config_entry.entry_id)]
107  )
108 
109 
110 class StateImageEntity(TemplateEntity, ImageEntity):
111  """Representation of a template image."""
112 
113  _attr_should_poll = False
114  _attr_image_url: str | None = None
115 
116  def __init__(
117  self,
118  hass: HomeAssistant,
119  config: dict[str, Any],
120  unique_id: str | None,
121  ) -> None:
122  """Initialize the image."""
123  TemplateEntity.__init__(self, hass, config=config, unique_id=unique_id)
124  ImageEntity.__init__(self, hass, config[CONF_VERIFY_SSL])
125  self._url_template_url_template = config[CONF_URL]
127  hass,
128  config.get(CONF_DEVICE_ID),
129  )
130 
131  @property
132  def entity_picture(self) -> str | None:
133  """Return entity picture."""
134  if self._entity_picture_template_entity_picture_template:
135  return TemplateEntity.entity_picture.__get__(self)
136  # mypy doesn't know about fget: https://github.com/python/mypy/issues/6185
137  return ImageEntity.entity_picture.fget(self) # type: ignore[attr-defined]
138 
139  @callback
140  def _update_url(self, result):
141  if isinstance(result, TemplateError):
142  self._attr_image_url_attr_image_url = None
143  return
144  self._attr_image_last_updated_attr_image_last_updated = dt_util.utcnow()
145  self._cached_image_cached_image = None
146  self._attr_image_url_attr_image_url = result
147 
148  @callback
149  def _async_setup_templates(self) -> None:
150  """Set up templates."""
151  self.add_template_attributeadd_template_attribute("_url", self._url_template_url_template, None, self._update_url_update_url)
152  super()._async_setup_templates()
153 
154 
155 class TriggerImageEntity(TriggerEntity, ImageEntity):
156  """Image entity based on trigger data."""
157 
158  _attr_image_url: str | None = None
159 
160  domain = IMAGE_DOMAIN
161  extra_template_keys = (CONF_URL,)
162 
163  def __init__(
164  self,
165  hass: HomeAssistant,
166  coordinator: TriggerUpdateCoordinator,
167  config: dict,
168  ) -> None:
169  """Initialize the entity."""
170  TriggerEntity.__init__(self, hass, coordinator, config)
171  ImageEntity.__init__(self, hass, config[CONF_VERIFY_SSL])
172 
173  @property
174  def entity_picture(self) -> str | None:
175  """Return entity picture."""
176  # mypy doesn't know about fget: https://github.com/python/mypy/issues/6185
177  if CONF_PICTURE in self._config:
178  return TriggerEntity.entity_picture.fget(self) # type: ignore[attr-defined]
179  return ImageEntity.entity_picture.fget(self) # type: ignore[attr-defined]
180 
181  @callback
182  def _process_data(self) -> None:
183  """Process new data."""
184  super()._process_data()
185  self._attr_image_last_updated_attr_image_last_updated = dt_util.utcnow()
186  self._cached_image_cached_image = None
187  self._attr_image_url_attr_image_url = self._rendered.get(CONF_URL)
None __init__(self, HomeAssistant hass, dict[str, Any] config, str|None unique_id)
Definition: image.py:121
None __init__(self, HomeAssistant hass, TriggerUpdateCoordinator coordinator, dict config)
Definition: image.py:168
None add_template_attribute(self, str attribute, Template template, Callable[[Any], Any]|None validator=None, Callable[[Any], None]|None on_update=None, bool none_on_template_error=False)
web.Response get(self, web.Request request, str config_key)
Definition: view.py:88
None async_setup_platform(HomeAssistant hass, ConfigType config, AddEntitiesCallback async_add_entities, DiscoveryInfoType|None discovery_info=None)
Definition: image.py:74
None async_setup_entry(HomeAssistant hass, ConfigEntry config_entry, AddEntitiesCallback async_add_entities)
Definition: image.py:100
list[StateImageEntity] _async_create_entities(HomeAssistant hass, list[dict[str, Any]] definitions, str|None unique_id_prefix)
Definition: image.py:58
vol.Schema make_template_entity_common_schema(str default_name)
dr.DeviceInfo|None async_device_info_to_link_from_device_id(HomeAssistant hass, str|None device_id)
Definition: device.py:44