rasa-pro 3.12.0.dev12__py3-none-any.whl → 3.12.0rc1__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.

Potentially problematic release.


This version of rasa-pro might be problematic. Click here for more details.

Files changed (153) hide show
  1. rasa/anonymization/anonymization_rule_executor.py +16 -10
  2. rasa/cli/data.py +16 -0
  3. rasa/cli/inspect.py +20 -1
  4. rasa/cli/project_templates/calm/config.yml +2 -2
  5. rasa/cli/project_templates/calm/endpoints.yml +2 -2
  6. rasa/cli/shell.py +3 -3
  7. rasa/cli/utils.py +12 -0
  8. rasa/core/actions/action.py +99 -193
  9. rasa/core/actions/action_handle_digressions.py +142 -0
  10. rasa/core/actions/action_run_slot_rejections.py +16 -4
  11. rasa/core/actions/forms.py +10 -5
  12. rasa/core/channels/__init__.py +4 -0
  13. rasa/core/channels/studio_chat.py +19 -0
  14. rasa/core/channels/telegram.py +42 -24
  15. rasa/core/channels/voice_ready/audiocodes.py +42 -23
  16. rasa/core/channels/voice_ready/utils.py +1 -1
  17. rasa/core/channels/voice_stream/asr/asr_engine.py +10 -4
  18. rasa/core/channels/voice_stream/asr/azure.py +14 -1
  19. rasa/core/channels/voice_stream/asr/deepgram.py +20 -4
  20. rasa/core/channels/voice_stream/audiocodes.py +264 -0
  21. rasa/core/channels/voice_stream/browser_audio.py +5 -1
  22. rasa/core/channels/voice_stream/call_state.py +10 -1
  23. rasa/core/channels/voice_stream/genesys.py +335 -0
  24. rasa/core/channels/voice_stream/tts/azure.py +11 -2
  25. rasa/core/channels/voice_stream/tts/cartesia.py +29 -10
  26. rasa/core/channels/voice_stream/twilio_media_streams.py +2 -1
  27. rasa/core/channels/voice_stream/voice_channel.py +25 -3
  28. rasa/core/constants.py +2 -0
  29. rasa/core/migrate.py +2 -2
  30. rasa/core/nlg/contextual_response_rephraser.py +18 -1
  31. rasa/core/nlg/generator.py +83 -15
  32. rasa/core/nlg/response.py +6 -3
  33. rasa/core/nlg/translate.py +55 -0
  34. rasa/core/policies/enterprise_search_prompt_with_citation_template.jinja2 +1 -1
  35. rasa/core/policies/flows/flow_executor.py +47 -46
  36. rasa/core/processor.py +72 -9
  37. rasa/core/run.py +4 -3
  38. rasa/dialogue_understanding/commands/can_not_handle_command.py +20 -2
  39. rasa/dialogue_understanding/commands/cancel_flow_command.py +80 -4
  40. rasa/dialogue_understanding/commands/change_flow_command.py +20 -2
  41. rasa/dialogue_understanding/commands/chit_chat_answer_command.py +20 -2
  42. rasa/dialogue_understanding/commands/clarify_command.py +29 -3
  43. rasa/dialogue_understanding/commands/command.py +1 -16
  44. rasa/dialogue_understanding/commands/command_syntax_manager.py +55 -0
  45. rasa/dialogue_understanding/commands/correct_slots_command.py +11 -2
  46. rasa/dialogue_understanding/commands/handle_digressions_command.py +150 -0
  47. rasa/dialogue_understanding/commands/human_handoff_command.py +20 -2
  48. rasa/dialogue_understanding/commands/knowledge_answer_command.py +20 -2
  49. rasa/dialogue_understanding/commands/prompt_command.py +94 -0
  50. rasa/dialogue_understanding/commands/repeat_bot_messages_command.py +20 -2
  51. rasa/dialogue_understanding/commands/set_slot_command.py +29 -15
  52. rasa/dialogue_understanding/commands/skip_question_command.py +20 -2
  53. rasa/dialogue_understanding/commands/start_flow_command.py +61 -2
  54. rasa/dialogue_understanding/commands/utils.py +98 -4
  55. rasa/dialogue_understanding/constants.py +1 -0
  56. rasa/dialogue_understanding/generator/__init__.py +2 -0
  57. rasa/dialogue_understanding/generator/command_generator.py +110 -73
  58. rasa/dialogue_understanding/generator/command_parser.py +16 -13
  59. rasa/dialogue_understanding/generator/constants.py +3 -0
  60. rasa/dialogue_understanding/generator/llm_based_command_generator.py +170 -5
  61. rasa/dialogue_understanding/generator/llm_command_generator.py +5 -3
  62. rasa/dialogue_understanding/generator/multi_step/multi_step_llm_command_generator.py +26 -4
  63. rasa/dialogue_understanding/generator/nlu_command_adapter.py +44 -3
  64. rasa/dialogue_understanding/generator/prompt_templates/__init__.py +0 -0
  65. rasa/dialogue_understanding/generator/prompt_templates/command_prompt_template.jinja2 +60 -0
  66. rasa/dialogue_understanding/generator/prompt_templates/command_prompt_v2_claude_3_5_sonnet_20240620_template.jinja2 +77 -0
  67. rasa/dialogue_understanding/generator/prompt_templates/command_prompt_v2_default.jinja2 +68 -0
  68. rasa/dialogue_understanding/generator/{single_step/command_prompt_template.jinja2 → prompt_templates/command_prompt_v2_gpt_4o_2024_11_20_template.jinja2} +1 -1
  69. rasa/dialogue_understanding/generator/single_step/compact_llm_command_generator.py +460 -0
  70. rasa/dialogue_understanding/generator/single_step/single_step_llm_command_generator.py +12 -318
  71. rasa/dialogue_understanding/generator/utils.py +32 -1
  72. rasa/dialogue_understanding/patterns/collect_information.py +1 -1
  73. rasa/dialogue_understanding/patterns/correction.py +13 -1
  74. rasa/dialogue_understanding/patterns/default_flows_for_patterns.yml +78 -2
  75. rasa/dialogue_understanding/patterns/handle_digressions.py +81 -0
  76. rasa/dialogue_understanding/patterns/validate_slot.py +65 -0
  77. rasa/dialogue_understanding/processor/command_processor.py +154 -28
  78. rasa/dialogue_understanding/utils.py +31 -0
  79. rasa/dialogue_understanding_test/README.md +50 -0
  80. rasa/dialogue_understanding_test/du_test_case.py +28 -8
  81. rasa/dialogue_understanding_test/du_test_result.py +13 -9
  82. rasa/dialogue_understanding_test/io.py +14 -0
  83. rasa/dialogue_understanding_test/test_case_simulation/test_case_tracker_simulator.py +3 -3
  84. rasa/e2e_test/utils/io.py +0 -37
  85. rasa/engine/graph.py +1 -0
  86. rasa/engine/language.py +140 -0
  87. rasa/engine/recipes/config_files/default_config.yml +4 -0
  88. rasa/engine/recipes/default_recipe.py +2 -0
  89. rasa/engine/recipes/graph_recipe.py +2 -0
  90. rasa/engine/storage/local_model_storage.py +1 -0
  91. rasa/engine/storage/storage.py +4 -1
  92. rasa/model_manager/runner_service.py +7 -4
  93. rasa/model_manager/socket_bridge.py +7 -6
  94. rasa/model_manager/warm_rasa_process.py +0 -1
  95. rasa/model_training.py +24 -27
  96. rasa/shared/constants.py +15 -13
  97. rasa/shared/core/constants.py +30 -3
  98. rasa/shared/core/domain.py +13 -20
  99. rasa/shared/core/events.py +13 -2
  100. rasa/shared/core/flows/constants.py +11 -0
  101. rasa/shared/core/flows/flow.py +100 -19
  102. rasa/shared/core/flows/flows_yaml_schema.json +69 -3
  103. rasa/shared/core/flows/steps/collect.py +19 -37
  104. rasa/shared/core/flows/utils.py +43 -4
  105. rasa/shared/core/flows/validation.py +1 -1
  106. rasa/shared/core/slot_mappings.py +350 -111
  107. rasa/shared/core/slots.py +154 -3
  108. rasa/shared/core/trackers.py +77 -2
  109. rasa/shared/importers/importer.py +50 -2
  110. rasa/shared/nlu/constants.py +1 -0
  111. rasa/shared/nlu/training_data/schemas/responses.yml +19 -12
  112. rasa/shared/providers/_configs/azure_entra_id_config.py +541 -0
  113. rasa/shared/providers/_configs/azure_openai_client_config.py +138 -3
  114. rasa/shared/providers/_configs/client_config.py +3 -1
  115. rasa/shared/providers/_configs/default_litellm_client_config.py +3 -1
  116. rasa/shared/providers/_configs/huggingface_local_embedding_client_config.py +3 -1
  117. rasa/shared/providers/_configs/litellm_router_client_config.py +3 -1
  118. rasa/shared/providers/_configs/model_group_config.py +4 -2
  119. rasa/shared/providers/_configs/oauth_config.py +33 -0
  120. rasa/shared/providers/_configs/openai_client_config.py +3 -1
  121. rasa/shared/providers/_configs/rasa_llm_client_config.py +3 -1
  122. rasa/shared/providers/_configs/self_hosted_llm_client_config.py +3 -1
  123. rasa/shared/providers/constants.py +6 -0
  124. rasa/shared/providers/embedding/azure_openai_embedding_client.py +28 -3
  125. rasa/shared/providers/embedding/litellm_router_embedding_client.py +3 -1
  126. rasa/shared/providers/llm/_base_litellm_client.py +42 -17
  127. rasa/shared/providers/llm/azure_openai_llm_client.py +81 -25
  128. rasa/shared/providers/llm/default_litellm_llm_client.py +3 -1
  129. rasa/shared/providers/llm/litellm_router_llm_client.py +29 -8
  130. rasa/shared/providers/llm/llm_client.py +23 -7
  131. rasa/shared/providers/llm/openai_llm_client.py +9 -3
  132. rasa/shared/providers/llm/rasa_llm_client.py +11 -2
  133. rasa/shared/providers/llm/self_hosted_llm_client.py +30 -11
  134. rasa/shared/providers/router/_base_litellm_router_client.py +3 -1
  135. rasa/shared/providers/router/router_client.py +3 -1
  136. rasa/shared/utils/constants.py +3 -0
  137. rasa/shared/utils/llm.py +31 -8
  138. rasa/shared/utils/pykwalify_extensions.py +24 -0
  139. rasa/shared/utils/schemas/domain.yml +26 -1
  140. rasa/telemetry.py +45 -14
  141. rasa/tracing/config.py +2 -0
  142. rasa/tracing/constants.py +12 -0
  143. rasa/tracing/instrumentation/instrumentation.py +36 -0
  144. rasa/tracing/instrumentation/metrics.py +41 -0
  145. rasa/tracing/metric_instrument_provider.py +40 -0
  146. rasa/utils/common.py +0 -1
  147. rasa/validator.py +561 -89
  148. rasa/version.py +1 -1
  149. {rasa_pro-3.12.0.dev12.dist-info → rasa_pro-3.12.0rc1.dist-info}/METADATA +2 -1
  150. {rasa_pro-3.12.0.dev12.dist-info → rasa_pro-3.12.0rc1.dist-info}/RECORD +153 -134
  151. {rasa_pro-3.12.0.dev12.dist-info → rasa_pro-3.12.0rc1.dist-info}/NOTICE +0 -0
  152. {rasa_pro-3.12.0.dev12.dist-info → rasa_pro-3.12.0rc1.dist-info}/WHEEL +0 -0
  153. {rasa_pro-3.12.0.dev12.dist-info → rasa_pro-3.12.0rc1.dist-info}/entry_points.txt +0 -0
rasa/shared/core/slots.py CHANGED
@@ -1,7 +1,12 @@
1
+ from __future__ import annotations
2
+
1
3
  import logging
2
4
  from abc import ABC, abstractmethod
5
+ from dataclasses import dataclass
3
6
  from typing import Any, Dict, List, Optional, Text, Type
4
7
 
8
+ from pydantic import BaseModel, Field
9
+
5
10
  import rasa.shared.core.constants
6
11
  import rasa.shared.utils.common
7
12
  import rasa.shared.utils.io
@@ -9,6 +14,9 @@ from rasa.shared.constants import (
9
14
  DOCS_URL_CATEGORICAL_SLOTS,
10
15
  DOCS_URL_NLU_BASED_SLOTS,
11
16
  DOCS_URL_SLOTS,
17
+ REFILL_UTTER,
18
+ REJECTIONS,
19
+ UTTER_ASK_PREFIX,
12
20
  )
13
21
  from rasa.shared.exceptions import RasaException
14
22
 
@@ -23,6 +31,80 @@ class InvalidSlotConfigError(RasaException, ValueError):
23
31
  """Raised if a slot's config is invalid."""
24
32
 
25
33
 
34
+ @dataclass
35
+ class SlotRejection:
36
+ """A pair of validation condition and an utterance for the case of failure."""
37
+
38
+ if_: str
39
+ """The condition that should be checked."""
40
+ utter: str
41
+ """The utterance that should be executed if the condition is met."""
42
+
43
+ @staticmethod
44
+ def from_dict(data: Dict[str, Any]) -> SlotRejection:
45
+ """Create a SlotRejection object from serialized data.
46
+
47
+ Args:
48
+ data: data for a SlotRejection object in a serialized format
49
+
50
+ Returns:
51
+ A SlotRejection object
52
+ """
53
+ return SlotRejection(
54
+ if_=data["if"],
55
+ utter=data["utter"],
56
+ )
57
+
58
+ def as_dict(self) -> Dict[str, Any]:
59
+ """Serialize the SlotRejection object.
60
+
61
+ Returns:
62
+ the SlotRejection object as serialized data
63
+ """
64
+ return {
65
+ "if": self.if_,
66
+ "utter": self.utter,
67
+ }
68
+
69
+
70
+ class SlotValidation(BaseModel):
71
+ rejections: List[SlotRejection] = Field(alias=REJECTIONS)
72
+ """how the slot value is validated using predicate evaluation."""
73
+ refill_utter: str = Field(alias=REFILL_UTTER)
74
+ """The utterance that the assistant uses to ask for the slot."""
75
+
76
+ @staticmethod
77
+ def from_dict(data: Dict[str, Any]) -> SlotValidation:
78
+ """Creates a SlotValidation object from serialised data.
79
+
80
+ Args:
81
+ data: data for a SlotValidation object in a serialized format
82
+
83
+ Returns:
84
+ A SlotValidation object
85
+ """
86
+ rejections = data.get(REJECTIONS)
87
+ if rejections is not None:
88
+ rejections = [
89
+ SlotRejection.from_dict(rejection) for rejection in rejections
90
+ ]
91
+
92
+ return SlotValidation(
93
+ rejections=rejections, refill_utter=data.get(REFILL_UTTER)
94
+ )
95
+
96
+ def as_dict(self) -> Dict[str, Any]:
97
+ """Serialize the SlotValidation object.
98
+
99
+ Returns:
100
+ the SlotValidation object as serialized data
101
+ """
102
+ return {
103
+ REJECTIONS: [rejection.as_dict() for rejection in self.rejections],
104
+ REFILL_UTTER: self.refill_utter,
105
+ }
106
+
107
+
26
108
  class Slot(ABC):
27
109
  """Key-value store for storing information during a conversation."""
28
110
 
@@ -41,6 +123,8 @@ class Slot(ABC):
41
123
  influence_conversation: bool = True,
42
124
  is_builtin: bool = False,
43
125
  shared_for_coexistence: bool = False,
126
+ filled_by: Optional[str] = None,
127
+ validation: Optional[Dict[str, Any]] = None,
44
128
  ) -> None:
45
129
  """Create a Slot.
46
130
 
@@ -57,9 +141,14 @@ class Slot(ABC):
57
141
  such as `return_value`.
58
142
  shared_for_coexistence: If `True` the slot is not forgotten after either
59
143
  dm1 or CALM finishes.
144
+ filled_by: The name of the extractor that fills the slot.
145
+ validation: The validation rules that should be used to validate
146
+ slot values.
60
147
  """
148
+ from rasa.shared.core.slot_mappings import SlotMapping
149
+
61
150
  self.name = name
62
- self.mappings = mappings
151
+ self.mappings = [SlotMapping.from_dict(mapping, name) for mapping in mappings]
63
152
  self._value = initial_value
64
153
  self.initial_value = initial_value
65
154
  self._value_reset_delay = value_reset_delay
@@ -67,6 +156,13 @@ class Slot(ABC):
67
156
  self._has_been_set = False
68
157
  self.is_builtin = is_builtin
69
158
  self.shared_for_coexistence = shared_for_coexistence
159
+ self._filled_by = filled_by
160
+
161
+ if validation:
162
+ validation.setdefault(REFILL_UTTER, f"{UTTER_ASK_PREFIX}{self.name}")
163
+ self.validation = (
164
+ SlotValidation.from_dict(validation) if validation else validation
165
+ )
70
166
 
71
167
  def feature_dimensionality(self) -> int:
72
168
  """How many features this single slot creates.
@@ -132,6 +228,16 @@ class Slot(ABC):
132
228
  self._value = value
133
229
  self._has_been_set = True
134
230
 
231
+ @property
232
+ def filled_by(self) -> Optional[str]:
233
+ """Gets the slot's latest value extractor."""
234
+ return self._filled_by
235
+
236
+ @filled_by.setter
237
+ def filled_by(self, extractor: str) -> None:
238
+ """Sets the slot's latest value extractor."""
239
+ self._filled_by = extractor
240
+
135
241
  def has_same_coerced_value(self, other_value: Any) -> bool:
136
242
  """Checks if the coerced value of is the same as the slot value.
137
243
 
@@ -176,12 +282,15 @@ class Slot(ABC):
176
282
 
177
283
  def persistence_info(self) -> Dict[str, Any]:
178
284
  """Returns relevant information to persist this slot."""
179
- return {
285
+ persistence_info_dict = {
180
286
  "type": rasa.shared.utils.common.module_path_from_instance(self),
181
287
  "initial_value": self.initial_value,
182
288
  "influence_conversation": self.influence_conversation,
183
- "mappings": self.mappings,
289
+ "mappings": [mapping.as_dict() for mapping in self.mappings],
184
290
  }
291
+ if self.validation:
292
+ persistence_info_dict["validation"] = self.validation.as_dict() # type: ignore
293
+ return persistence_info_dict
185
294
 
186
295
  def fingerprint(self) -> Text:
187
296
  """Returns a unique hash for the slot which is stable across python runs.
@@ -198,6 +307,10 @@ class Slot(ABC):
198
307
  return False
199
308
  return self.name == other.name and self.value == other.value
200
309
 
310
+ def requires_validation(self) -> bool:
311
+ """Indicates if the slot requires validation."""
312
+ return True if self.validation else False
313
+
201
314
 
202
315
  class FloatSlot(Slot):
203
316
  """A slot storing a float value."""
@@ -215,6 +328,7 @@ class FloatSlot(Slot):
215
328
  influence_conversation: bool = True,
216
329
  is_builtin: bool = False,
217
330
  shared_for_coexistence: bool = False,
331
+ filled_by: Optional[str] = None,
218
332
  ) -> None:
219
333
  """Creates a FloatSlot.
220
334
 
@@ -230,6 +344,7 @@ class FloatSlot(Slot):
230
344
  influence_conversation,
231
345
  is_builtin,
232
346
  shared_for_coexistence,
347
+ filled_by=filled_by,
233
348
  )
234
349
  self.max_value = max_value
235
350
  self.min_value = min_value
@@ -387,6 +502,7 @@ class CategoricalSlot(Slot):
387
502
  influence_conversation: bool = True,
388
503
  is_builtin: bool = False,
389
504
  shared_for_coexistence: bool = False,
505
+ filled_by: Optional[str] = None,
390
506
  ) -> None:
391
507
  """Creates a `Categorical Slot` (see parent class for detailed docstring)."""
392
508
  super().__init__(
@@ -397,6 +513,7 @@ class CategoricalSlot(Slot):
397
513
  influence_conversation,
398
514
  is_builtin,
399
515
  shared_for_coexistence,
516
+ filled_by=filled_by,
400
517
  )
401
518
  if values and None in values:
402
519
  rasa.shared.utils.io.raise_warning(
@@ -607,6 +724,7 @@ class AnySlot(Slot):
607
724
  influence_conversation: bool = False,
608
725
  is_builtin: bool = False,
609
726
  shared_for_coexistence: bool = False,
727
+ filled_by: Optional[str] = None,
610
728
  ) -> None:
611
729
  """Creates an `Any Slot` (see parent class for detailed docstring).
612
730
 
@@ -630,6 +748,7 @@ class AnySlot(Slot):
630
748
  influence_conversation,
631
749
  is_builtin,
632
750
  shared_for_coexistence,
751
+ filled_by=filled_by,
633
752
  )
634
753
 
635
754
  def __eq__(self, other: Any) -> bool:
@@ -652,3 +771,35 @@ class AnySlot(Slot):
652
771
  f"implement a custom slot type by subclassing '{Slot.__name__}'. "
653
772
  f"See the documentation for more information: {DOCS_URL_NLU_BASED_SLOTS}"
654
773
  )
774
+
775
+
776
+ class StrictCategoricalSlot(CategoricalSlot):
777
+ """A categorical slot that strictly enforces allowed values."""
778
+
779
+ type_name = "strict_categorical"
780
+
781
+ def coerce_value(self, value: Any) -> Any:
782
+ """Coerce the value to one of the allowed ones or raise an error if invalid."""
783
+ if value is None:
784
+ return value
785
+
786
+ for allowed_value in self.values:
787
+ # Allowed values are always stored as strings, so we can use casefold().
788
+ if value.casefold() == allowed_value.casefold():
789
+ return allowed_value
790
+
791
+ raise InvalidSlotConfigError(
792
+ f"Value '{value}' is not allowed for the slot '{self.name}'. "
793
+ f"Allowed values are: {self.values}"
794
+ )
795
+
796
+ @Slot.value.setter # type: ignore[attr-defined,misc]
797
+ def value(self, value: Any) -> None:
798
+ """Set the slot's value using strict coercion."""
799
+ coerced_value = self.coerce_value(value)
800
+ super(StrictCategoricalSlot, self.__class__).value.fset(self, coerced_value)
801
+
802
+ def add_default_value(self) -> None:
803
+ # StrictCategoricalSlot enforces validation against a specified set of values,
804
+ # so default values should not be automatically added.
805
+ pass
@@ -6,6 +6,7 @@ import os
6
6
  import time
7
7
  from collections import deque
8
8
  from enum import Enum
9
+ from functools import cached_property
9
10
  from typing import (
10
11
  TYPE_CHECKING,
11
12
  Any,
@@ -26,6 +27,7 @@ from typing import (
26
27
  )
27
28
 
28
29
  import rasa.shared.utils.io
30
+ from rasa.engine.language import Language
29
31
  from rasa.shared.constants import (
30
32
  ASSISTANT_ID_KEY,
31
33
  DEFAULT_SENDER_ID,
@@ -37,6 +39,7 @@ from rasa.shared.core.constants import (
37
39
  ACTION_SESSION_START_NAME,
38
40
  ACTIVE_LOOP,
39
41
  FOLLOWUP_ACTION,
42
+ LANGUAGE_SLOT,
40
43
  LOOP_NAME,
41
44
  PREVIOUS_ACTION,
42
45
  SHOULD_NOT_BE_SET,
@@ -61,7 +64,8 @@ from rasa.shared.core.events import (
61
64
  UserUttered,
62
65
  )
63
66
  from rasa.shared.core.flows import FlowsList
64
- from rasa.shared.core.slots import AnySlot, Slot
67
+ from rasa.shared.core.slots import AnySlot, Slot, StrictCategoricalSlot
68
+ from rasa.shared.exceptions import RasaException
65
69
  from rasa.shared.nlu.constants import (
66
70
  ACTION_NAME,
67
71
  ACTION_TEXT,
@@ -916,11 +920,13 @@ class DialogueStateTracker:
916
920
  continue
917
921
  slot.reset()
918
922
 
919
- def _set_slot(self, key: Text, value: Any) -> None:
923
+ def _set_slot(self, key: Text, value: Any, filled_by: Optional[str] = None) -> None:
920
924
  """Sets the value of a slot if that slot exists."""
921
925
  if key in self.slots:
922
926
  slot = self.slots[key]
923
927
  slot.value = value
928
+ if filled_by is not None:
929
+ slot.filled_by = filled_by
924
930
  else:
925
931
  logger.error(
926
932
  f"Tried to set non existent slot '{key}'. Make sure you "
@@ -1095,6 +1101,75 @@ class DialogueStateTracker:
1095
1101
 
1096
1102
  return FlowsList(active_flows)
1097
1103
 
1104
+ @cached_property
1105
+ def supported_languages(self) -> List[Language]:
1106
+ """Returns the supported languages for this model configuration
1107
+
1108
+ Returns:
1109
+ A list of supported languages.
1110
+ """
1111
+ if LANGUAGE_SLOT not in self.slots:
1112
+ raise RasaException(
1113
+ f"The required slot '{LANGUAGE_SLOT}' is missing from the tracker. "
1114
+ f"Please ensure that a slot named '{LANGUAGE_SLOT}' exists."
1115
+ )
1116
+
1117
+ language_slot = self.slots[LANGUAGE_SLOT]
1118
+
1119
+ if not isinstance(language_slot, StrictCategoricalSlot):
1120
+ raise RasaException(
1121
+ f"The slot '{LANGUAGE_SLOT}' must be of type "
1122
+ f"'{StrictCategoricalSlot.type_name}'. "
1123
+ f"Please update the slot configuration accordingly."
1124
+ )
1125
+
1126
+ return [
1127
+ Language.from_language_code(language_code)
1128
+ for language_code in language_slot.values
1129
+ ]
1130
+
1131
+ @property
1132
+ def current_language(self) -> Optional[Language]:
1133
+ """Get the language of the current conversation.
1134
+
1135
+ Returns:
1136
+ The language of the current conversation or `None` if not set.
1137
+ """
1138
+ language_code = self.get_slot("language")
1139
+ if not language_code:
1140
+ return None
1141
+
1142
+ supported_languages = self.supported_languages or []
1143
+ matching_language = (
1144
+ language
1145
+ for language in supported_languages
1146
+ if language.code == language_code
1147
+ )
1148
+ return next(matching_language, None)
1149
+
1150
+ @property
1151
+ def default_language(self) -> Language:
1152
+ """Get the assistant's default language.
1153
+
1154
+ Returns:
1155
+ The assistant's default language.
1156
+
1157
+ Raises:
1158
+ RasaException: If no default language is defined in the config.
1159
+ """
1160
+ supported_languages = self.supported_languages or []
1161
+ matching_language = (
1162
+ language for language in supported_languages if language.is_default is True
1163
+ )
1164
+ try:
1165
+ return next(matching_language)
1166
+ except StopIteration:
1167
+ raise RasaException(
1168
+ "No default language configured. "
1169
+ "Please configure the `language` parameter in config.yml file. "
1170
+ "Example: `language: en`."
1171
+ )
1172
+
1098
1173
 
1099
1174
  class TrackerEventDiffEngine:
1100
1175
  """Computes event difference of two trackers."""
@@ -20,16 +20,19 @@ import rasa.shared.constants
20
20
  import rasa.shared.core.constants
21
21
  import rasa.shared.utils.common
22
22
  import rasa.shared.utils.io
23
+ from rasa.shared.constants import CONFIG_ADDITIONAL_LANGUAGES_KEY, CONFIG_LANGUAGE_KEY
23
24
  from rasa.shared.core.domain import (
24
25
  IS_RETRIEVAL_INTENT_KEY,
25
26
  KEY_ACTIONS,
26
27
  KEY_E2E_ACTIONS,
27
28
  KEY_INTENTS,
28
29
  KEY_RESPONSES,
30
+ KEY_SLOTS,
29
31
  Domain,
30
32
  )
31
33
  from rasa.shared.core.events import ActionExecuted, UserUttered
32
34
  from rasa.shared.core.flows import FlowsList
35
+ from rasa.shared.core.slots import StrictCategoricalSlot
33
36
  from rasa.shared.core.training_data.structures import StoryGraph
34
37
  from rasa.shared.nlu.constants import ACTION_NAME, ENTITIES
35
38
  from rasa.shared.nlu.training_data.message import Message
@@ -202,8 +205,10 @@ class TrainingDataImporter(ABC):
202
205
  )
203
206
  ]
204
207
 
205
- return E2EImporter(
206
- FlowSyncImporter(ResponsesSyncImporter(CombinedDataImporter(importers)))
208
+ return LanguageImporter(
209
+ E2EImporter(
210
+ FlowSyncImporter(ResponsesSyncImporter(CombinedDataImporter(importers)))
211
+ )
207
212
  )
208
213
 
209
214
  @staticmethod
@@ -522,6 +527,49 @@ class FlowSyncImporter(PassThroughImporter):
522
527
  return self._importer.get_domain()
523
528
 
524
529
 
530
+ class LanguageImporter(PassThroughImporter):
531
+ """Importer that configures the language settings in the domain."""
532
+
533
+ @cached_method
534
+ def get_domain(self) -> Domain:
535
+ domain = self._importer.get_domain()
536
+ if domain.is_empty():
537
+ return domain
538
+
539
+ config = self._importer.get_config()
540
+ language = config.get(CONFIG_LANGUAGE_KEY)
541
+ additional_languages = config.get(CONFIG_ADDITIONAL_LANGUAGES_KEY) or []
542
+
543
+ values = additional_languages.copy()
544
+ if language and language not in values:
545
+ values.append(language)
546
+
547
+ # Prepare the serialized representation of the language slot
548
+ slot_name = rasa.shared.core.constants.LANGUAGE_SLOT
549
+ serialized_slot: Dict[Text, Any] = {
550
+ "type": StrictCategoricalSlot.type_name,
551
+ "initial_value": language,
552
+ "values": values,
553
+ "mappings": [],
554
+ "is_builtin": True,
555
+ }
556
+
557
+ domain_with_language_slot = Domain.from_dict(
558
+ {KEY_SLOTS: {slot_name: serialized_slot}}
559
+ )
560
+ return domain.merge(domain_with_language_slot)
561
+
562
+ @cached_method
563
+ def get_user_domain(self) -> Domain:
564
+ """Delegate to the underlying importer to get the user domain."""
565
+ return self._importer.get_user_domain()
566
+
567
+ @cached_method
568
+ def get_user_flows(self) -> FlowsList:
569
+ """Delegate to the underlying importer to get user flows."""
570
+ return self._importer.get_user_flows()
571
+
572
+
525
573
  class ResponsesSyncImporter(PassThroughImporter):
526
574
  """Importer that syncs `responses` between Domain and NLU training data.
527
575
 
@@ -55,3 +55,4 @@ SPLIT_ENTITIES_BY_COMMA_DEFAULT_VALUE = True
55
55
  SINGLE_ENTITY_ALLOWED_INTERLEAVING_CHARSET = {".", ",", " ", ";"}
56
56
 
57
57
  SET_SLOT_COMMAND = "set slot"
58
+ HANDLE_DIGRESSIONS_COMMAND = "handle digressions"
@@ -17,6 +17,12 @@ schema;responses:
17
17
  required: False
18
18
  text:
19
19
  type: "str"
20
+ translation:
21
+ type: "map"
22
+ allowempty: True
23
+ mapping:
24
+ regex;(.*):
25
+ type: "str"
20
26
  image:
21
27
  type: "str"
22
28
  custom:
@@ -32,6 +38,18 @@ schema;responses:
32
38
  type: "str"
33
39
  payload:
34
40
  type: "str"
41
+ translation:
42
+ type: "map"
43
+ allowempty: True
44
+ mapping:
45
+ regex;(.*):
46
+ type: "map"
47
+ allowempty: True
48
+ mapping:
49
+ title:
50
+ type: "str"
51
+ payload:
52
+ type: "str"
35
53
  button_type:
36
54
  type: "str"
37
55
  quick_replies:
@@ -57,15 +75,4 @@ schema;responses:
57
75
  metadata:
58
76
  type: "any"
59
77
  condition:
60
- type: "seq"
61
- sequence:
62
- - type: "map"
63
- allowempty: False
64
- mapping:
65
- type:
66
- type: "str"
67
- enum: ['slot']
68
- name:
69
- type: "str"
70
- value:
71
- type: "any"
78
+ type: "any"