@@ -223,22 +223,22 @@ async def async_process(self, user_input: ConversationInput) -> ConversationResu
223
223
# Check if a trigger matched
224
224
if isinstance (result , SentenceTriggerResult ):
225
225
# Gather callback responses in parallel
226
- trigger_responses = await asyncio .gather (
227
- * (
228
- self ._trigger_sentences [trigger_id ].callback (
229
- result .sentence , trigger_result
230
- )
231
- for trigger_id , trigger_result in result .matched_triggers .items ()
226
+ trigger_callbacks = [
227
+ self ._trigger_sentences [trigger_id ].callback (
228
+ result .sentence , trigger_result
232
229
)
233
- )
230
+ for trigger_id , trigger_result in result .matched_triggers .items ()
231
+ ]
234
232
235
233
# Use last non-empty result as response.
236
234
#
237
235
# There may be multiple copies of a trigger running when editing in
238
236
# the UI, so it's critical that we filter out empty responses here.
239
237
response_text : str | None = None
240
- for trigger_response in trigger_responses :
241
- response_text = response_text or trigger_response
238
+ for trigger_future in asyncio .as_completed (trigger_callbacks ):
239
+ if trigger_response := await trigger_future :
240
+ response_text = trigger_response
241
+ break
242
242
243
243
# Convert to conversation result
244
244
response = intent .IntentResponse (language = language )
@@ -316,6 +316,20 @@ async def async_process(self, user_input: ConversationInput) -> ConversationResu
316
316
),
317
317
conversation_id ,
318
318
)
319
+ except intent .DuplicateNamesMatchedError as duplicate_names_error :
320
+ # Intent was valid, but two or more entities with the same name matched.
321
+ (
322
+ error_response_type ,
323
+ error_response_args ,
324
+ ) = _get_duplicate_names_matched_response (duplicate_names_error )
325
+ return _make_error_result (
326
+ language ,
327
+ intent .IntentResponseErrorCode .NO_VALID_TARGETS ,
328
+ self ._get_error_text (
329
+ error_response_type , lang_intents , ** error_response_args
330
+ ),
331
+ conversation_id ,
332
+ )
319
333
except intent .IntentHandleError :
320
334
# Intent was valid and entities matched constraints, but an error
321
335
# occurred during handling.
@@ -724,7 +738,12 @@ def _make_slot_lists(self) -> dict[str, SlotList]:
724
738
if async_should_expose (self .hass , DOMAIN , state .entity_id )
725
739
]
726
740
727
- # Gather exposed entity names
741
+ # Gather exposed entity names.
742
+ #
743
+ # NOTE: We do not pass entity ids in here because multiple entities may
744
+ # have the same name. The intent matcher doesn't gather all matching
745
+ # values for a list, just the first. So we will need to match by name no
746
+ # matter what.
728
747
entity_names = []
729
748
for state in states :
730
749
# Checked against "requires_context" and "excludes_context" in hassil
@@ -740,20 +759,23 @@ def _make_slot_lists(self) -> dict[str, SlotList]:
740
759
741
760
if not entity :
742
761
# Default name
743
- entity_names .append ((state .name , state .entity_id , context ))
762
+ entity_names .append ((state .name , state .name , context ))
744
763
continue
745
764
746
765
if entity .aliases :
747
766
for alias in entity .aliases :
748
767
if not alias .strip ():
749
768
continue
750
769
751
- entity_names .append ((alias , state . entity_id , context ))
770
+ entity_names .append ((alias , alias , context ))
752
771
753
772
# Default name
754
- entity_names .append ((state .name , state .entity_id , context ))
773
+ entity_names .append ((state .name , state .name , context ))
755
774
756
- # Expose all areas
775
+ # Expose all areas.
776
+ #
777
+ # We pass in area id here with the expectation that no two areas will
778
+ # share the same name or alias.
757
779
areas = ar .async_get (self .hass )
758
780
area_names = []
759
781
for area in areas .async_list_areas ():
@@ -984,6 +1006,20 @@ def _get_no_states_matched_response(
984
1006
return ErrorKey .NO_INTENT , {}
985
1007
986
1008
1009
+ def _get_duplicate_names_matched_response (
1010
+ duplicate_names_error : intent .DuplicateNamesMatchedError ,
1011
+ ) -> tuple [ErrorKey , dict [str , Any ]]:
1012
+ """Return key and template arguments for error when intent returns duplicate matches."""
1013
+
1014
+ if duplicate_names_error .area :
1015
+ return ErrorKey .DUPLICATE_ENTITIES_IN_AREA , {
1016
+ "entity" : duplicate_names_error .name ,
1017
+ "area" : duplicate_names_error .area ,
1018
+ }
1019
+
1020
+ return ErrorKey .DUPLICATE_ENTITIES , {"entity" : duplicate_names_error .name }
1021
+
1022
+
987
1023
def _collect_list_references (expression : Expression , list_names : set [str ]) -> None :
988
1024
"""Collect list reference names recursively."""
989
1025
if isinstance (expression , Sequence ):
0 commit comments