From 222330e7c50788cc92e47387775ad1dc784c5c7b Mon Sep 17 00:00:00 2001 From: Michael Hansen Date: Wed, 12 Mar 2025 16:54:54 -0500 Subject: [PATCH 1/4] Add language scores websocket command --- homeassistant/components/conversation/http.py | 18 ++ .../conversation/snapshots/test_http.ambr | 299 ++++++++++++++++++ tests/components/conversation/test_http.py | 22 ++ 3 files changed, 339 insertions(+) diff --git a/homeassistant/components/conversation/http.py b/homeassistant/components/conversation/http.py index 4d8526a4fd4..8a61d41a42a 100644 --- a/homeassistant/components/conversation/http.py +++ b/homeassistant/components/conversation/http.py @@ -3,11 +3,13 @@ from __future__ import annotations from collections.abc import Iterable +from dataclasses import asdict from typing import Any from aiohttp import web from hassil.recognize import MISSING_ENTITY, RecognizeResult from hassil.string_matcher import UnmatchedRangeEntity, UnmatchedTextEntity +from home_assistant_intents import get_language_scores import voluptuous as vol from homeassistant.components import http, websocket_api @@ -38,6 +40,7 @@ def async_setup(hass: HomeAssistant) -> None: websocket_api.async_register_command(hass, websocket_list_agents) websocket_api.async_register_command(hass, websocket_list_sentences) websocket_api.async_register_command(hass, websocket_hass_agent_debug) + websocket_api.async_register_command(hass, websocket_hass_agent_language_scores) @websocket_api.websocket_command( @@ -336,6 +339,21 @@ def _get_unmatched_slots( return unmatched_slots +@websocket_api.websocket_command( + {vol.Required("type"): "conversation/agent/homeassistant/language_scores"} +) +@websocket_api.async_response +async def websocket_hass_agent_language_scores( + hass: HomeAssistant, + connection: websocket_api.ActiveConnection, + msg: dict[str, Any], +) -> None: + """Get support scores per language.""" + scores = await hass.async_add_executor_job(get_language_scores) + result = {lang_key: asdict(lang_scores) for lang_key, lang_scores in scores.items()} + connection.send_result(msg["id"], result) + + class ConversationProcessView(http.HomeAssistantView): """View to process text.""" diff --git a/tests/components/conversation/snapshots/test_http.ambr b/tests/components/conversation/snapshots/test_http.ambr index abce735dd8a..944394ccc64 100644 --- a/tests/components/conversation/snapshots/test_http.ambr +++ b/tests/components/conversation/snapshots/test_http.ambr @@ -729,3 +729,302 @@ ]), }) # --- +# name: test_ws_hass_language_scores + dict({ + 'af_ZA': dict({ + 'cloud': 0, + 'focused_local': 0, + 'full_local': 0, + }), + 'ar_JO': dict({ + 'cloud': 1, + 'focused_local': 0, + 'full_local': 0, + }), + 'bg_BG': dict({ + 'cloud': 3, + 'focused_local': 0, + 'full_local': 0, + }), + 'bn_BD': dict({ + 'cloud': 0, + 'focused_local': 0, + 'full_local': 0, + }), + 'bn_IN': dict({ + 'cloud': 1, + 'focused_local': 0, + 'full_local': 0, + }), + 'ca_ES': dict({ + 'cloud': 3, + 'focused_local': 0, + 'full_local': 0, + }), + 'cs_CZ': dict({ + 'cloud': 1, + 'focused_local': 0, + 'full_local': 0, + }), + 'da_DK': dict({ + 'cloud': 3, + 'focused_local': 0, + 'full_local': 0, + }), + 'de_CH': dict({ + 'cloud': 3, + 'focused_local': 0, + 'full_local': 0, + }), + 'de_DE': dict({ + 'cloud': 3, + 'focused_local': 2, + 'full_local': 3, + }), + 'el_GR': dict({ + 'cloud': 1, + 'focused_local': 0, + 'full_local': 0, + }), + 'en_US': dict({ + 'cloud': 3, + 'focused_local': 2, + 'full_local': 3, + }), + 'es_MX': dict({ + 'cloud': 3, + 'focused_local': 2, + 'full_local': 3, + }), + 'et_EE': dict({ + 'cloud': 1, + 'focused_local': 0, + 'full_local': 0, + }), + 'eu_ES': dict({ + 'cloud': 1, + 'focused_local': 0, + 'full_local': 0, + }), + 'fa_IR': dict({ + 'cloud': 1, + 'focused_local': 0, + 'full_local': 0, + }), + 'fi_FI': dict({ + 'cloud': 3, + 'focused_local': 0, + 'full_local': 0, + }), + 'fr_FR': dict({ + 'cloud': 3, + 'focused_local': 2, + 'full_local': 0, + }), + 'gl_ES': dict({ + 'cloud': 1, + 'focused_local': 0, + 'full_local': 0, + }), + 'gu_IN': dict({ + 'cloud': 0, + 'focused_local': 0, + 'full_local': 0, + }), + 'he_IL': dict({ + 'cloud': 1, + 'focused_local': 0, + 'full_local': 0, + }), + 'hi_IN': dict({ + 'cloud': 1, + 'focused_local': 0, + 'full_local': 0, + }), + 'hr_HR': dict({ + 'cloud': 3, + 'focused_local': 0, + 'full_local': 0, + }), + 'hu_HU': dict({ + 'cloud': 3, + 'focused_local': 0, + 'full_local': 0, + }), + 'id_ID': dict({ + 'cloud': 1, + 'focused_local': 0, + 'full_local': 0, + }), + 'is_IS': dict({ + 'cloud': 1, + 'focused_local': 0, + 'full_local': 0, + }), + 'it_IT': dict({ + 'cloud': 3, + 'focused_local': 2, + 'full_local': 3, + }), + 'ka_GE': dict({ + 'cloud': 1, + 'focused_local': 0, + 'full_local': 0, + }), + 'kn_IN': dict({ + 'cloud': 0, + 'focused_local': 0, + 'full_local': 0, + }), + 'ko_KR': dict({ + 'cloud': 1, + 'focused_local': 0, + 'full_local': 0, + }), + 'lb_LU': dict({ + 'cloud': 0, + 'focused_local': 0, + 'full_local': 0, + }), + 'lt_LT': dict({ + 'cloud': 1, + 'focused_local': 0, + 'full_local': 0, + }), + 'lv_LV': dict({ + 'cloud': 1, + 'focused_local': 0, + 'full_local': 0, + }), + 'ml_IN': dict({ + 'cloud': 1, + 'focused_local': 0, + 'full_local': 0, + }), + 'mn_MN': dict({ + 'cloud': 1, + 'focused_local': 0, + 'full_local': 0, + }), + 'mr_IN': dict({ + 'cloud': 0, + 'focused_local': 0, + 'full_local': 0, + }), + 'ms_MY': dict({ + 'cloud': 1, + 'focused_local': 0, + 'full_local': 0, + }), + 'nb_NO': dict({ + 'cloud': 3, + 'focused_local': 0, + 'full_local': 0, + }), + 'ne_NP': dict({ + 'cloud': 1, + 'focused_local': 0, + 'full_local': 0, + }), + 'nl_BE': dict({ + 'cloud': 3, + 'focused_local': 0, + 'full_local': 0, + }), + 'nl_NL': dict({ + 'cloud': 3, + 'focused_local': 2, + 'full_local': 0, + }), + 'pl_PL': dict({ + 'cloud': 3, + 'focused_local': 0, + 'full_local': 0, + }), + 'pt_BR': dict({ + 'cloud': 3, + 'focused_local': 0, + 'full_local': 3, + }), + 'pt_PT': dict({ + 'cloud': 3, + 'focused_local': 0, + 'full_local': 3, + }), + 'ro_RO': dict({ + 'cloud': 3, + 'focused_local': 0, + 'full_local': 0, + }), + 'ru_RU': dict({ + 'cloud': 1, + 'focused_local': 0, + 'full_local': 0, + }), + 'sk_SK': dict({ + 'cloud': 1, + 'focused_local': 0, + 'full_local': 0, + }), + 'sl_SI': dict({ + 'cloud': 3, + 'focused_local': 0, + 'full_local': 0, + }), + 'sr_RS': dict({ + 'cloud': 3, + 'focused_local': 0, + 'full_local': 0, + }), + 'sv_SE': dict({ + 'cloud': 3, + 'focused_local': 0, + 'full_local': 0, + }), + 'sw_KE': dict({ + 'cloud': 1, + 'focused_local': 0, + 'full_local': 0, + }), + 'te_IN': dict({ + 'cloud': 1, + 'focused_local': 0, + 'full_local': 0, + }), + 'th_TH': dict({ + 'cloud': 1, + 'focused_local': 0, + 'full_local': 0, + }), + 'tr_TR': dict({ + 'cloud': 1, + 'focused_local': 0, + 'full_local': 0, + }), + 'uk_UA': dict({ + 'cloud': 1, + 'focused_local': 0, + 'full_local': 0, + }), + 'ur_IN': dict({ + 'cloud': 0, + 'focused_local': 0, + 'full_local': 0, + }), + 'vi_VN': dict({ + 'cloud': 1, + 'focused_local': 0, + 'full_local': 0, + }), + 'zh_CN': dict({ + 'cloud': 1, + 'focused_local': 0, + 'full_local': 0, + }), + 'zh_HK': dict({ + 'cloud': 3, + 'focused_local': 0, + 'full_local': 0, + }), + }) +# --- diff --git a/tests/components/conversation/test_http.py b/tests/components/conversation/test_http.py index 6d69ec3c739..b003e478312 100644 --- a/tests/components/conversation/test_http.py +++ b/tests/components/conversation/test_http.py @@ -536,3 +536,25 @@ async def test_ws_hass_agent_debug_sentence_trigger( # Trigger should not have been executed assert len(calls) == 0 + + +async def test_ws_hass_language_scores( + hass: HomeAssistant, + init_components, + hass_ws_client: WebSocketGenerator, + snapshot: SnapshotAssertion, +) -> None: + """Test getting language support scores.""" + client = await hass_ws_client(hass) + + await client.send_json_auto_id( + {"type": "conversation/agent/homeassistant/language_scores"} + ) + + msg = await client.receive_json() + + assert msg["success"] + assert msg["result"] == snapshot + + # Sanity check + assert msg["result"]["en_US"] == {"cloud": 3, "focused_local": 2, "full_local": 3} From c2932debdeaf61f233410a108800282db7d6bf54 Mon Sep 17 00:00:00 2001 From: Michael Hansen Date: Wed, 12 Mar 2025 17:46:37 -0500 Subject: [PATCH 2/4] Don't store language scores in snapshot --- .../conversation/snapshots/test_http.ambr | 299 ------------------ tests/components/conversation/test_http.py | 1 - 2 files changed, 300 deletions(-) diff --git a/tests/components/conversation/snapshots/test_http.ambr b/tests/components/conversation/snapshots/test_http.ambr index 944394ccc64..abce735dd8a 100644 --- a/tests/components/conversation/snapshots/test_http.ambr +++ b/tests/components/conversation/snapshots/test_http.ambr @@ -729,302 +729,3 @@ ]), }) # --- -# name: test_ws_hass_language_scores - dict({ - 'af_ZA': dict({ - 'cloud': 0, - 'focused_local': 0, - 'full_local': 0, - }), - 'ar_JO': dict({ - 'cloud': 1, - 'focused_local': 0, - 'full_local': 0, - }), - 'bg_BG': dict({ - 'cloud': 3, - 'focused_local': 0, - 'full_local': 0, - }), - 'bn_BD': dict({ - 'cloud': 0, - 'focused_local': 0, - 'full_local': 0, - }), - 'bn_IN': dict({ - 'cloud': 1, - 'focused_local': 0, - 'full_local': 0, - }), - 'ca_ES': dict({ - 'cloud': 3, - 'focused_local': 0, - 'full_local': 0, - }), - 'cs_CZ': dict({ - 'cloud': 1, - 'focused_local': 0, - 'full_local': 0, - }), - 'da_DK': dict({ - 'cloud': 3, - 'focused_local': 0, - 'full_local': 0, - }), - 'de_CH': dict({ - 'cloud': 3, - 'focused_local': 0, - 'full_local': 0, - }), - 'de_DE': dict({ - 'cloud': 3, - 'focused_local': 2, - 'full_local': 3, - }), - 'el_GR': dict({ - 'cloud': 1, - 'focused_local': 0, - 'full_local': 0, - }), - 'en_US': dict({ - 'cloud': 3, - 'focused_local': 2, - 'full_local': 3, - }), - 'es_MX': dict({ - 'cloud': 3, - 'focused_local': 2, - 'full_local': 3, - }), - 'et_EE': dict({ - 'cloud': 1, - 'focused_local': 0, - 'full_local': 0, - }), - 'eu_ES': dict({ - 'cloud': 1, - 'focused_local': 0, - 'full_local': 0, - }), - 'fa_IR': dict({ - 'cloud': 1, - 'focused_local': 0, - 'full_local': 0, - }), - 'fi_FI': dict({ - 'cloud': 3, - 'focused_local': 0, - 'full_local': 0, - }), - 'fr_FR': dict({ - 'cloud': 3, - 'focused_local': 2, - 'full_local': 0, - }), - 'gl_ES': dict({ - 'cloud': 1, - 'focused_local': 0, - 'full_local': 0, - }), - 'gu_IN': dict({ - 'cloud': 0, - 'focused_local': 0, - 'full_local': 0, - }), - 'he_IL': dict({ - 'cloud': 1, - 'focused_local': 0, - 'full_local': 0, - }), - 'hi_IN': dict({ - 'cloud': 1, - 'focused_local': 0, - 'full_local': 0, - }), - 'hr_HR': dict({ - 'cloud': 3, - 'focused_local': 0, - 'full_local': 0, - }), - 'hu_HU': dict({ - 'cloud': 3, - 'focused_local': 0, - 'full_local': 0, - }), - 'id_ID': dict({ - 'cloud': 1, - 'focused_local': 0, - 'full_local': 0, - }), - 'is_IS': dict({ - 'cloud': 1, - 'focused_local': 0, - 'full_local': 0, - }), - 'it_IT': dict({ - 'cloud': 3, - 'focused_local': 2, - 'full_local': 3, - }), - 'ka_GE': dict({ - 'cloud': 1, - 'focused_local': 0, - 'full_local': 0, - }), - 'kn_IN': dict({ - 'cloud': 0, - 'focused_local': 0, - 'full_local': 0, - }), - 'ko_KR': dict({ - 'cloud': 1, - 'focused_local': 0, - 'full_local': 0, - }), - 'lb_LU': dict({ - 'cloud': 0, - 'focused_local': 0, - 'full_local': 0, - }), - 'lt_LT': dict({ - 'cloud': 1, - 'focused_local': 0, - 'full_local': 0, - }), - 'lv_LV': dict({ - 'cloud': 1, - 'focused_local': 0, - 'full_local': 0, - }), - 'ml_IN': dict({ - 'cloud': 1, - 'focused_local': 0, - 'full_local': 0, - }), - 'mn_MN': dict({ - 'cloud': 1, - 'focused_local': 0, - 'full_local': 0, - }), - 'mr_IN': dict({ - 'cloud': 0, - 'focused_local': 0, - 'full_local': 0, - }), - 'ms_MY': dict({ - 'cloud': 1, - 'focused_local': 0, - 'full_local': 0, - }), - 'nb_NO': dict({ - 'cloud': 3, - 'focused_local': 0, - 'full_local': 0, - }), - 'ne_NP': dict({ - 'cloud': 1, - 'focused_local': 0, - 'full_local': 0, - }), - 'nl_BE': dict({ - 'cloud': 3, - 'focused_local': 0, - 'full_local': 0, - }), - 'nl_NL': dict({ - 'cloud': 3, - 'focused_local': 2, - 'full_local': 0, - }), - 'pl_PL': dict({ - 'cloud': 3, - 'focused_local': 0, - 'full_local': 0, - }), - 'pt_BR': dict({ - 'cloud': 3, - 'focused_local': 0, - 'full_local': 3, - }), - 'pt_PT': dict({ - 'cloud': 3, - 'focused_local': 0, - 'full_local': 3, - }), - 'ro_RO': dict({ - 'cloud': 3, - 'focused_local': 0, - 'full_local': 0, - }), - 'ru_RU': dict({ - 'cloud': 1, - 'focused_local': 0, - 'full_local': 0, - }), - 'sk_SK': dict({ - 'cloud': 1, - 'focused_local': 0, - 'full_local': 0, - }), - 'sl_SI': dict({ - 'cloud': 3, - 'focused_local': 0, - 'full_local': 0, - }), - 'sr_RS': dict({ - 'cloud': 3, - 'focused_local': 0, - 'full_local': 0, - }), - 'sv_SE': dict({ - 'cloud': 3, - 'focused_local': 0, - 'full_local': 0, - }), - 'sw_KE': dict({ - 'cloud': 1, - 'focused_local': 0, - 'full_local': 0, - }), - 'te_IN': dict({ - 'cloud': 1, - 'focused_local': 0, - 'full_local': 0, - }), - 'th_TH': dict({ - 'cloud': 1, - 'focused_local': 0, - 'full_local': 0, - }), - 'tr_TR': dict({ - 'cloud': 1, - 'focused_local': 0, - 'full_local': 0, - }), - 'uk_UA': dict({ - 'cloud': 1, - 'focused_local': 0, - 'full_local': 0, - }), - 'ur_IN': dict({ - 'cloud': 0, - 'focused_local': 0, - 'full_local': 0, - }), - 'vi_VN': dict({ - 'cloud': 1, - 'focused_local': 0, - 'full_local': 0, - }), - 'zh_CN': dict({ - 'cloud': 1, - 'focused_local': 0, - 'full_local': 0, - }), - 'zh_HK': dict({ - 'cloud': 3, - 'focused_local': 0, - 'full_local': 0, - }), - }) -# --- diff --git a/tests/components/conversation/test_http.py b/tests/components/conversation/test_http.py index b003e478312..ec70c97f912 100644 --- a/tests/components/conversation/test_http.py +++ b/tests/components/conversation/test_http.py @@ -554,7 +554,6 @@ async def test_ws_hass_language_scores( msg = await client.receive_json() assert msg["success"] - assert msg["result"] == snapshot # Sanity check assert msg["result"]["en_US"] == {"cloud": 3, "focused_local": 2, "full_local": 3} From 6a3ea036890d0e1ed137ba581aeecad24d9c801d Mon Sep 17 00:00:00 2001 From: Michael Hansen Date: Mon, 24 Mar 2025 14:31:10 -0500 Subject: [PATCH 3/4] Add language/country args for preferred lang --- homeassistant/components/conversation/http.py | 19 +++++++- tests/components/conversation/test_http.py | 48 ++++++++++++++++--- 2 files changed, 59 insertions(+), 8 deletions(-) diff --git a/homeassistant/components/conversation/http.py b/homeassistant/components/conversation/http.py index 8a61d41a42a..efcdcb8d69b 100644 --- a/homeassistant/components/conversation/http.py +++ b/homeassistant/components/conversation/http.py @@ -340,7 +340,11 @@ def _get_unmatched_slots( @websocket_api.websocket_command( - {vol.Required("type"): "conversation/agent/homeassistant/language_scores"} + { + vol.Required("type"): "conversation/agent/homeassistant/language_scores", + vol.Optional("language"): str, + vol.Optional("country"): str, + } ) @websocket_api.async_response async def websocket_hass_agent_language_scores( @@ -349,8 +353,19 @@ async def websocket_hass_agent_language_scores( msg: dict[str, Any], ) -> None: """Get support scores per language.""" + language = msg.get("language", hass.config.language) + country = msg.get("country", hass.config.country) + scores = await hass.async_add_executor_job(get_language_scores) - result = {lang_key: asdict(lang_scores) for lang_key, lang_scores in scores.items()} + matching_langs = language_util.matches(language, scores.keys(), country=country) + preferred_lang = matching_langs[0] if matching_langs else language + result = { + "languages": { + lang_key: asdict(lang_scores) for lang_key, lang_scores in scores.items() + }, + "preferred_language": preferred_lang, + } + connection.send_result(msg["id"], result) diff --git a/tests/components/conversation/test_http.py b/tests/components/conversation/test_http.py index ec70c97f912..77fa97ad845 100644 --- a/tests/components/conversation/test_http.py +++ b/tests/components/conversation/test_http.py @@ -539,10 +539,7 @@ async def test_ws_hass_agent_debug_sentence_trigger( async def test_ws_hass_language_scores( - hass: HomeAssistant, - init_components, - hass_ws_client: WebSocketGenerator, - snapshot: SnapshotAssertion, + hass: HomeAssistant, init_components, hass_ws_client: WebSocketGenerator ) -> None: """Test getting language support scores.""" client = await hass_ws_client(hass) @@ -552,8 +549,47 @@ async def test_ws_hass_language_scores( ) msg = await client.receive_json() - assert msg["success"] # Sanity check - assert msg["result"]["en_US"] == {"cloud": 3, "focused_local": 2, "full_local": 3} + result = msg["result"] + assert result["languages"]["en-US"] == { + "cloud": 3, + "focused_local": 2, + "full_local": 3, + } + + +async def test_ws_hass_language_scores_with_filter( + hass: HomeAssistant, init_components, hass_ws_client: WebSocketGenerator +) -> None: + """Test getting language support scores with language/country filter.""" + client = await hass_ws_client(hass) + + # Language filter + await client.send_json_auto_id( + {"type": "conversation/agent/homeassistant/language_scores", "language": "de"} + ) + + msg = await client.receive_json() + assert msg["success"] + + # German should be preferred + result = msg["result"] + assert result["preferred_language"] == "de-DE" + + # Language/country filter + await client.send_json_auto_id( + { + "type": "conversation/agent/homeassistant/language_scores", + "language": "en", + "country": "GB", + } + ) + + msg = await client.receive_json() + assert msg["success"] + + # GB English should be preferred + result = msg["result"] + assert result["preferred_language"] == "en-GB" From 865e4b1603029dbae5d703c9188746530a8b8c3a Mon Sep 17 00:00:00 2001 From: Michael Hansen Date: Mon, 24 Mar 2025 14:33:00 -0500 Subject: [PATCH 4/4] Bump intents to 2025.3.24 for dash lang code --- homeassistant/components/conversation/manifest.json | 2 +- homeassistant/package_constraints.txt | 2 +- requirements_all.txt | 2 +- requirements_test_all.txt | 2 +- script/hassfest/docker/Dockerfile | 2 +- 5 files changed, 5 insertions(+), 5 deletions(-) diff --git a/homeassistant/components/conversation/manifest.json b/homeassistant/components/conversation/manifest.json index 56d5e28e642..acaa2ef0967 100644 --- a/homeassistant/components/conversation/manifest.json +++ b/homeassistant/components/conversation/manifest.json @@ -6,5 +6,5 @@ "documentation": "https://www.home-assistant.io/integrations/conversation", "integration_type": "system", "quality_scale": "internal", - "requirements": ["hassil==2.2.3", "home-assistant-intents==2025.3.23"] + "requirements": ["hassil==2.2.3", "home-assistant-intents==2025.3.24"] } diff --git a/homeassistant/package_constraints.txt b/homeassistant/package_constraints.txt index d85bf08338b..471571c18a6 100644 --- a/homeassistant/package_constraints.txt +++ b/homeassistant/package_constraints.txt @@ -39,7 +39,7 @@ hass-nabucasa==0.94.0 hassil==2.2.3 home-assistant-bluetooth==1.13.1 home-assistant-frontend==20250306.0 -home-assistant-intents==2025.3.23 +home-assistant-intents==2025.3.24 httpx==0.28.1 ifaddr==0.2.0 Jinja2==3.1.6 diff --git a/requirements_all.txt b/requirements_all.txt index d59c11f5709..812708bb4cb 100644 --- a/requirements_all.txt +++ b/requirements_all.txt @@ -1158,7 +1158,7 @@ holidays==0.68 home-assistant-frontend==20250306.0 # homeassistant.components.conversation -home-assistant-intents==2025.3.23 +home-assistant-intents==2025.3.24 # homeassistant.components.homematicip_cloud homematicip==1.1.7 diff --git a/requirements_test_all.txt b/requirements_test_all.txt index 00706fc3c57..08385274dea 100644 --- a/requirements_test_all.txt +++ b/requirements_test_all.txt @@ -984,7 +984,7 @@ holidays==0.68 home-assistant-frontend==20250306.0 # homeassistant.components.conversation -home-assistant-intents==2025.3.23 +home-assistant-intents==2025.3.24 # homeassistant.components.homematicip_cloud homematicip==1.1.7 diff --git a/script/hassfest/docker/Dockerfile b/script/hassfest/docker/Dockerfile index c4f66faafb0..5524a70f1c2 100644 --- a/script/hassfest/docker/Dockerfile +++ b/script/hassfest/docker/Dockerfile @@ -25,7 +25,7 @@ RUN --mount=from=ghcr.io/astral-sh/uv:0.6.8,source=/uv,target=/bin/uv \ -c /usr/src/homeassistant/homeassistant/package_constraints.txt \ -r /usr/src/homeassistant/requirements.txt \ stdlib-list==0.10.0 pipdeptree==2.25.1 tqdm==4.67.1 ruff==0.11.0 \ - PyTurboJPEG==1.7.5 go2rtc-client==0.1.2 ha-ffmpeg==3.2.2 hassil==2.2.3 home-assistant-intents==2025.3.23 mutagen==1.47.0 pymicro-vad==1.0.1 pyspeex-noise==1.0.2 + PyTurboJPEG==1.7.5 go2rtc-client==0.1.2 ha-ffmpeg==3.2.2 hassil==2.2.3 home-assistant-intents==2025.3.24 mutagen==1.47.0 pymicro-vad==1.0.1 pyspeex-noise==1.0.2 LABEL "name"="hassfest" LABEL "maintainer"="Home Assistant "