Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add ServiceValidationError and translation support #1961

Merged
Merged
Show file tree
Hide file tree
Changes from 4 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
21 changes: 21 additions & 0 deletions blog/2023-10-29-service-exceptions-and-translations.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
---
jbouwh marked this conversation as resolved.
Show resolved Hide resolved
author: Jan Bouwhuis
authorURL: https://github.com/jbouwh
authorImageURL: https://avatars.githubusercontent.com/u/7188918?s=96&v=4
title: Exception handling during and translation support
---

## Exception handling during service calls

Currently service calls that raise exceptions will log full stack traces. Service calls that fail due to invalid user input don't need a stack trace but would benefit from a helpful error message in the users own language.

To be able to suppress the stack trace in these cases, we introduce `ServiceValidationError` as a new exception type. The `ServiceValidationError` exception can be raised instead of `HomeAssistantError` during the execution of a service call. The error message is to be shown to through the UI, and in the logs. The stack trace is printed at debug level, so it is still possible to print it to support development. In case an other exception is raised during a service (including `HomeAssistantError`) nothing changes and a full stack trace will be printed. [Read more](/docs/core/platform/handling_exceptions).
jbouwh marked this conversation as resolved.
Show resolved Hide resolved

## Translation support for Exceptions

The `HomeAssistantError` exception and it's sub classes, including `ServiceValidationError`, now accept a translation key to allow localization. [Read more](/docs/internationalization/core/#exceptions). The translation key will be used in cases where the frontend receives information about the exception.
jbouwh marked this conversation as resolved.
Show resolved Hide resolved

### Background

- Background [discussion](https://github.com/home-assistant/architecture/discussions/992).
- Implementation [Core PR #102592](https://github.com/home-assistant/core/pull/102592).
25 changes: 25 additions & 0 deletions docs/api/websocket.md
Original file line number Diff line number Diff line change
Expand Up @@ -525,3 +525,28 @@ If an error occurs, the `success` key in the `result` message will be set to `fa
}
}
```

### Error handling during service calls and translations

When a `ServiceValidationError` (`service_validation_error`) exception was raised during a websocket service call. A stack trace is printed to the logs at debug level only. Raising `HomeAssistantError` (`home_assistant_error`) or subclass exception will still print a full stack trace to the logs.
jbouwh marked this conversation as resolved.
Show resolved Hide resolved

During a websocket service calls translations are supported for `HomeAssistantError` and its subclasses. When a `translation_key` is set by the caller of the exception this, the keys `translation_domain` and `translation_placeholders` will be added to the error response.
jbouwh marked this conversation as resolved.
Show resolved Hide resolved

```json
{
"id": 24,
"type":"result",
"success": false,
"error": {
"code": "service_validation_error",
"message": "Option 'custom' is not a supported mode.",
"translation_key": "unsupported_mode",
"translation_domain": "kitchen_sink",
"translation_placeholders": {
"mode": "custom"
}
}
}
```

[Read more](/docs/core/platform/handling_exceptions) about handling exceptions or and the [localization of exceptions](/docs/internationalization/core/#exceptions).
14 changes: 14 additions & 0 deletions docs/core/platform/handling_exceptions.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,14 @@
---
title: "Handling Exceptions"
jbouwh marked this conversation as resolved.
Show resolved Hide resolved
---




Operations like service calls and entity methods (e.g. *Set HVAC Mode*) should have proper exception handling. Raise `ServiceValidationError` on an invalid user input and raise `HomeAssistantError` for other failures such as a problem communicating with a device. Note that the exception stack trace will be logged.

Raise `ServiceValidation` for validation error occurs during a service calls where printing a full stack trace to the logs is not warranted. This exceptions class will only log exception stack traces at debug level.

## Localizing Exceptions

Home Assistant [supports localization](/docs/internationalization/core/#exceptions) for `HomeAssistantError` and it's subclasses like `ServiceValidationError`.
2 changes: 1 addition & 1 deletion docs/integration_quality_scale_index.md
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,7 @@ This integration is able to cope when things go wrong. It will not print any exc
- Handles expiration of auth credentials. Refresh if possible or print correct error and fail setup. If based on a config entry, should trigger a new config entry flow to re-authorize. ([docs](config_entries_config_flow_handler.md#reauthentication))
- Handles internet unavailable. Log a warning once when unavailable, log once when reconnected.
- Handles device/service unavailable. Log a warning once when unavailable, log once when reconnected.
- Operations like service calls and entity methods (e.g. *Set HVAC Mode*) have proper exception handling. Raise `ValueError` on invalid user input and raise `HomeAssistantError` for other failures such as a problem communicating with a device.
- Operations like service calls and entity methods (e.g. *Set HVAC Mode*) have proper exception handling. Raise `ServiceValidationError` on invalid user input and raise `HomeAssistantError` for other failures such as a problem communicating with a device. [Read more](/docs/core/platform/handling_exceptions) about handling exceptions.
- Set `available` property to `False` if appropriate ([docs](core/entity.md#generic-properties))
- Entities have unique ID (if available) ([docs](entity_registry_index.md#unique-id-requirements))

Expand Down
37 changes: 37 additions & 0 deletions docs/internationalization/core.md
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@ The `strings.json` contains translations for different things that the integrati
| `title` | Title of the integration. |
| `config` | Translations for the config flow. |
| `device_automation` | Translations for device automations. |
| `exceptions` | Translations for for error messages. |
jbouwh marked this conversation as resolved.
Show resolved Hide resolved
| `issues` | Translations for repairs issues. |
| `options` | Translations for the options flow. |
| `selectors` | Selectors of the integration. |
Expand Down Expand Up @@ -159,6 +160,42 @@ The translation strings for device automations are defined under the `device_aut

```

### Exceptions

Localization is supported for `HomeAssistantError` and its subclasses.
The translation strings for exceptions are defined under the `exception` key. An example strings file below describes the different supported keys.
jbouwh marked this conversation as resolved.
Show resolved Hide resolved

```json
{
"exceptions": {
// Translations for known exceptions
"invalid_index": {
"message": "Invalid index selected, expected [0,1,2]. Got {index}"
}
}
}

```

Example of raising exception with localization during a service call:
jbouwh marked this conversation as resolved.
Show resolved Hide resolved

```python
async def async_select_index(hass: HomeAssistant, index: int) -> None:
"""Setup the config entry for my device."""
try:
check_index(index)
except ValueError as ex:
raise ServiceValidationError(
message,
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@jbouwh where does this variable come from.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Good point. It should be a static string with a message to be logged.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

translation_domain=DOMAIN,
translation_key="invalid_index",
translation_placeholders={
"index": index,
},
) from ex
```


### Issues

The translation strings for repairs issues are defined under the `issues` key. An example strings file below describes the different supported keys.
Expand Down
1 change: 1 addition & 0 deletions sidebars.js
Original file line number Diff line number Diff line change
Expand Up @@ -218,6 +218,7 @@ module.exports = {
"creating_integration_brand",
"core/platform/application_credentials",
"core/platform/backup",
"core/platform/handling_exceptions",
"core/platform/repairs",
"core/platform/reproduce_state",
"core/platform/significant_change",
Expand Down