2017-04-30 05:04:49 +00:00
|
|
|
"""The exceptions used by Home Assistant."""
|
2021-03-17 16:34:55 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2021-04-20 15:40:41 +00:00
|
|
|
from collections.abc import Generator, Sequence
|
|
|
|
from typing import TYPE_CHECKING
|
2021-02-21 13:54:36 +00:00
|
|
|
|
|
|
|
import attr
|
2019-12-09 15:42:10 +00:00
|
|
|
|
2018-11-21 11:26:08 +00:00
|
|
|
if TYPE_CHECKING:
|
2021-03-02 08:02:04 +00:00
|
|
|
from .core import Context
|
2018-11-21 11:26:08 +00:00
|
|
|
|
2021-08-16 21:12:06 +00:00
|
|
|
# mypy: disallow-any-generics
|
|
|
|
|
2015-08-30 02:34:35 +00:00
|
|
|
|
2015-08-30 01:11:24 +00:00
|
|
|
class HomeAssistantError(Exception):
|
2016-03-07 23:06:04 +00:00
|
|
|
"""General Home Assistant exception occurred."""
|
|
|
|
|
2015-08-30 01:11:24 +00:00
|
|
|
|
|
|
|
class InvalidEntityFormatError(HomeAssistantError):
|
2016-03-07 23:06:04 +00:00
|
|
|
"""When an invalid formatted entity is encountered."""
|
|
|
|
|
2015-08-30 01:11:24 +00:00
|
|
|
|
|
|
|
class NoEntitySpecifiedError(HomeAssistantError):
|
2016-03-07 23:06:04 +00:00
|
|
|
"""When no entity is specified."""
|
|
|
|
|
2015-12-12 03:07:03 +00:00
|
|
|
|
|
|
|
class TemplateError(HomeAssistantError):
|
2016-03-07 23:06:04 +00:00
|
|
|
"""Error during template rendering."""
|
|
|
|
|
2020-10-12 14:38:24 +00:00
|
|
|
def __init__(self, exception: Exception) -> None:
|
2017-04-30 05:04:49 +00:00
|
|
|
"""Init the error."""
|
2019-08-23 16:53:33 +00:00
|
|
|
super().__init__(f"{exception.__class__.__name__}: {exception}")
|
2017-06-26 16:41:48 +00:00
|
|
|
|
|
|
|
|
2021-02-21 13:54:36 +00:00
|
|
|
@attr.s
|
2021-02-08 09:47:57 +00:00
|
|
|
class ConditionError(HomeAssistantError):
|
|
|
|
"""Error during condition evaluation."""
|
|
|
|
|
2021-02-21 13:54:36 +00:00
|
|
|
# The type of the failed condition, such as 'and' or 'numeric_state'
|
|
|
|
type: str = attr.ib()
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def _indent(indent: int, message: str) -> str:
|
|
|
|
"""Return indentation."""
|
|
|
|
return " " * indent + message
|
|
|
|
|
2021-08-16 21:12:06 +00:00
|
|
|
def output(self, indent: int) -> Generator[str, None, None]:
|
2021-02-21 13:54:36 +00:00
|
|
|
"""Yield an indented representation."""
|
|
|
|
raise NotImplementedError()
|
|
|
|
|
|
|
|
def __str__(self) -> str:
|
|
|
|
"""Return string representation."""
|
|
|
|
return "\n".join(list(self.output(indent=0)))
|
|
|
|
|
|
|
|
|
|
|
|
@attr.s
|
|
|
|
class ConditionErrorMessage(ConditionError):
|
|
|
|
"""Condition error message."""
|
|
|
|
|
|
|
|
# A message describing this error
|
|
|
|
message: str = attr.ib()
|
|
|
|
|
2021-08-16 21:12:06 +00:00
|
|
|
def output(self, indent: int) -> Generator[str, None, None]:
|
2021-02-21 13:54:36 +00:00
|
|
|
"""Yield an indented representation."""
|
|
|
|
yield self._indent(indent, f"In '{self.type}' condition: {self.message}")
|
|
|
|
|
|
|
|
|
|
|
|
@attr.s
|
|
|
|
class ConditionErrorIndex(ConditionError):
|
|
|
|
"""Condition error with index."""
|
|
|
|
|
|
|
|
# The zero-based index of the failed condition, for conditions with multiple parts
|
|
|
|
index: int = attr.ib()
|
|
|
|
# The total number of parts in this condition, including non-failed parts
|
|
|
|
total: int = attr.ib()
|
|
|
|
# The error that this error wraps
|
|
|
|
error: ConditionError = attr.ib()
|
|
|
|
|
2021-08-16 21:12:06 +00:00
|
|
|
def output(self, indent: int) -> Generator[str, None, None]:
|
2021-02-21 13:54:36 +00:00
|
|
|
"""Yield an indented representation."""
|
|
|
|
if self.total > 1:
|
|
|
|
yield self._indent(
|
|
|
|
indent, f"In '{self.type}' (item {self.index+1} of {self.total}):"
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
yield self._indent(indent, f"In '{self.type}':")
|
|
|
|
|
|
|
|
yield from self.error.output(indent + 1)
|
|
|
|
|
|
|
|
|
|
|
|
@attr.s
|
|
|
|
class ConditionErrorContainer(ConditionError):
|
2021-02-22 07:11:59 +00:00
|
|
|
"""Condition error with subconditions."""
|
2021-02-21 13:54:36 +00:00
|
|
|
|
|
|
|
# List of ConditionErrors that this error wraps
|
|
|
|
errors: Sequence[ConditionError] = attr.ib()
|
|
|
|
|
2021-08-16 21:12:06 +00:00
|
|
|
def output(self, indent: int) -> Generator[str, None, None]:
|
2021-02-21 13:54:36 +00:00
|
|
|
"""Yield an indented representation."""
|
|
|
|
for item in self.errors:
|
|
|
|
yield from item.output(indent)
|
|
|
|
|
2021-02-08 09:47:57 +00:00
|
|
|
|
2021-04-10 05:41:29 +00:00
|
|
|
class IntegrationError(HomeAssistantError):
|
|
|
|
"""Base class for platform and config entry exceptions."""
|
|
|
|
|
|
|
|
def __str__(self) -> str:
|
|
|
|
"""Return a human readable error."""
|
|
|
|
return super().__str__() or str(self.__cause__)
|
|
|
|
|
|
|
|
|
|
|
|
class PlatformNotReady(IntegrationError):
|
2017-06-26 16:41:48 +00:00
|
|
|
"""Error to indicate that platform is not ready."""
|
|
|
|
|
2017-10-25 16:05:30 +00:00
|
|
|
|
2021-04-10 05:41:29 +00:00
|
|
|
class ConfigEntryNotReady(IntegrationError):
|
2018-10-04 13:53:50 +00:00
|
|
|
"""Error to indicate that config entry is not ready."""
|
|
|
|
|
|
|
|
|
2021-04-10 05:41:29 +00:00
|
|
|
class ConfigEntryAuthFailed(IntegrationError):
|
|
|
|
"""Error to indicate that config entry could not authenticate."""
|
|
|
|
|
|
|
|
|
2017-10-25 16:05:30 +00:00
|
|
|
class InvalidStateError(HomeAssistantError):
|
|
|
|
"""When an invalid state is encountered."""
|
|
|
|
|
2018-11-21 11:26:08 +00:00
|
|
|
|
|
|
|
class Unauthorized(HomeAssistantError):
|
|
|
|
"""When an action is unauthorized."""
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
def __init__(
|
|
|
|
self,
|
2021-03-22 12:29:39 +00:00
|
|
|
context: Context | None = None,
|
2021-03-17 16:34:55 +00:00
|
|
|
user_id: str | None = None,
|
|
|
|
entity_id: str | None = None,
|
|
|
|
config_entry_id: str | None = None,
|
|
|
|
perm_category: str | None = None,
|
|
|
|
permission: str | None = None,
|
2019-07-31 19:25:30 +00:00
|
|
|
) -> None:
|
2018-11-21 11:26:08 +00:00
|
|
|
"""Unauthorized error."""
|
|
|
|
super().__init__(self.__class__.__name__)
|
|
|
|
self.context = context
|
2020-08-19 12:57:38 +00:00
|
|
|
|
|
|
|
if user_id is None and context is not None:
|
|
|
|
user_id = context.user_id
|
|
|
|
|
2018-11-21 11:26:08 +00:00
|
|
|
self.user_id = user_id
|
|
|
|
self.entity_id = entity_id
|
2018-12-13 14:30:20 +00:00
|
|
|
self.config_entry_id = config_entry_id
|
|
|
|
# Not all actions have an ID (like adding config entry)
|
|
|
|
# We then use this fallback to know what category was unauth
|
|
|
|
self.perm_category = perm_category
|
2018-11-21 11:26:08 +00:00
|
|
|
self.permission = permission
|
|
|
|
|
|
|
|
|
|
|
|
class UnknownUser(Unauthorized):
|
|
|
|
"""When call is made with user ID that doesn't exist."""
|
2018-11-30 20:28:35 +00:00
|
|
|
|
|
|
|
|
|
|
|
class ServiceNotFound(HomeAssistantError):
|
|
|
|
"""Raised when a service is not found."""
|
|
|
|
|
|
|
|
def __init__(self, domain: str, service: str) -> None:
|
|
|
|
"""Initialize error."""
|
2019-08-23 16:53:33 +00:00
|
|
|
super().__init__(self, f"Service {domain}.{service} not found")
|
2018-11-30 20:28:35 +00:00
|
|
|
self.domain = domain
|
|
|
|
self.service = service
|
2019-05-14 05:09:11 +00:00
|
|
|
|
|
|
|
def __str__(self) -> str:
|
|
|
|
"""Return string representation."""
|
2020-12-02 09:32:25 +00:00
|
|
|
return f"Unable to find service {self.domain}.{self.service}"
|
2021-04-08 18:46:28 +00:00
|
|
|
|
|
|
|
|
|
|
|
class MaxLengthExceeded(HomeAssistantError):
|
|
|
|
"""Raised when a property value has exceeded the max character length."""
|
|
|
|
|
|
|
|
def __init__(self, value: str, property_name: str, max_length: int) -> None:
|
|
|
|
"""Initialize error."""
|
|
|
|
super().__init__(
|
|
|
|
self,
|
|
|
|
(
|
|
|
|
f"Value {value} for property {property_name} has a max length of "
|
|
|
|
f"{max_length} characters"
|
|
|
|
),
|
|
|
|
)
|
|
|
|
self.value = value
|
|
|
|
self.property_name = property_name
|
|
|
|
self.max_length = max_length
|
2021-04-13 12:18:51 +00:00
|
|
|
|
|
|
|
|
|
|
|
class RequiredParameterMissing(HomeAssistantError):
|
|
|
|
"""Raised when a required parameter is missing from a function call."""
|
|
|
|
|
|
|
|
def __init__(self, parameter_names: list[str]) -> None:
|
|
|
|
"""Initialize error."""
|
|
|
|
super().__init__(
|
|
|
|
self,
|
|
|
|
(
|
|
|
|
"Call must include at least one of the following parameters: "
|
|
|
|
f"{', '.join(parameter_names)}"
|
|
|
|
),
|
|
|
|
)
|
|
|
|
self.parameter_names = parameter_names
|