rasa-pro 3.12.0.dev2__py3-none-any.whl → 3.12.0.dev4__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 (502) hide show
  1. rasa/__main__.py +7 -7
  2. rasa/anonymization/anonymisation_rule_yaml_reader.py +1 -1
  3. rasa/anonymization/anonymization_pipeline.py +3 -3
  4. rasa/anonymization/anonymization_rule_executor.py +1 -1
  5. rasa/anonymization/anonymization_rule_orchestrator.py +2 -3
  6. rasa/cli/arguments/data.py +2 -2
  7. rasa/cli/arguments/evaluate.py +2 -1
  8. rasa/cli/arguments/interactive.py +1 -1
  9. rasa/cli/arguments/run.py +1 -1
  10. rasa/cli/arguments/test.py +7 -5
  11. rasa/cli/arguments/train.py +3 -3
  12. rasa/cli/arguments/visualize.py +2 -2
  13. rasa/cli/arguments/x.py +1 -0
  14. rasa/cli/data.py +4 -3
  15. rasa/cli/dialogue_understanding_test.py +121 -18
  16. rasa/cli/evaluate.py +1 -1
  17. rasa/cli/export.py +6 -6
  18. rasa/cli/interactive.py +4 -5
  19. rasa/cli/llm_fine_tuning.py +5 -5
  20. rasa/cli/markers.py +1 -2
  21. rasa/cli/project_templates/calm/actions/add_contact.py +1 -1
  22. rasa/cli/project_templates/tutorial/actions/actions.py +3 -2
  23. rasa/cli/shell.py +2 -3
  24. rasa/cli/studio/download.py +1 -2
  25. rasa/cli/studio/studio.py +2 -3
  26. rasa/cli/studio/train.py +0 -1
  27. rasa/cli/telemetry.py +2 -2
  28. rasa/cli/test.py +11 -11
  29. rasa/cli/utils.py +7 -5
  30. rasa/core/__init__.py +0 -1
  31. rasa/core/actions/action.py +42 -21
  32. rasa/core/actions/action_hangup.py +1 -1
  33. rasa/core/actions/action_repeat_bot_messages.py +2 -2
  34. rasa/core/actions/action_run_slot_rejections.py +2 -2
  35. rasa/core/actions/action_trigger_chitchat.py +1 -1
  36. rasa/core/actions/action_trigger_flow.py +5 -5
  37. rasa/core/actions/action_trigger_search.py +1 -1
  38. rasa/core/actions/forms.py +14 -12
  39. rasa/core/actions/http_custom_action_executor.py +8 -1
  40. rasa/core/actions/loops.py +3 -3
  41. rasa/core/actions/two_stage_fallback.py +13 -13
  42. rasa/core/auth_retry_tracker_store.py +1 -2
  43. rasa/core/brokers/broker.py +2 -1
  44. rasa/core/brokers/file.py +1 -1
  45. rasa/core/brokers/kafka.py +8 -8
  46. rasa/core/brokers/pika.py +8 -9
  47. rasa/core/brokers/sql.py +4 -3
  48. rasa/core/channels/__init__.py +3 -0
  49. rasa/core/channels/botframework.py +2 -2
  50. rasa/core/channels/callback.py +4 -4
  51. rasa/core/channels/channel.py +11 -11
  52. rasa/core/channels/console.py +0 -1
  53. rasa/core/channels/development_inspector.py +6 -6
  54. rasa/core/channels/facebook.py +5 -5
  55. rasa/core/channels/hangouts.py +7 -8
  56. rasa/core/channels/inspector/dist/assets/{arc-861ddd57.js → arc-632a63ec.js} +1 -1
  57. rasa/core/channels/inspector/dist/assets/{c4Diagram-d0fbc5ce-921f02db.js → c4Diagram-d0fbc5ce-081e0df4.js} +1 -1
  58. rasa/core/channels/inspector/dist/assets/{classDiagram-936ed81e-b436c4f8.js → classDiagram-936ed81e-3df0afc2.js} +1 -1
  59. rasa/core/channels/inspector/dist/assets/{classDiagram-v2-c3cb15f1-511a23cb.js → classDiagram-v2-c3cb15f1-8c5ed31e.js} +1 -1
  60. rasa/core/channels/inspector/dist/assets/{createText-62fc7601-ef476ecd.js → createText-62fc7601-89c73b31.js} +1 -1
  61. rasa/core/channels/inspector/dist/assets/{edges-f2ad444c-f1878e0a.js → edges-f2ad444c-4fc48c3e.js} +1 -1
  62. rasa/core/channels/inspector/dist/assets/{erDiagram-9d236eb7-fac75185.js → erDiagram-9d236eb7-907e0440.js} +1 -1
  63. rasa/core/channels/inspector/dist/assets/{flowDb-1972c806-201c5bbc.js → flowDb-1972c806-9ec53a3c.js} +1 -1
  64. rasa/core/channels/inspector/dist/assets/{flowDiagram-7ea5b25a-f904ae41.js → flowDiagram-7ea5b25a-41da787a.js} +1 -1
  65. rasa/core/channels/inspector/dist/assets/flowDiagram-v2-855bc5b3-8bea338b.js +1 -0
  66. rasa/core/channels/inspector/dist/assets/{flowchart-elk-definition-abe16c3d-1813da66.js → flowchart-elk-definition-abe16c3d-ce370633.js} +1 -1
  67. rasa/core/channels/inspector/dist/assets/{ganttDiagram-9b5ea136-872af172.js → ganttDiagram-9b5ea136-90a36523.js} +1 -1
  68. rasa/core/channels/inspector/dist/assets/{gitGraphDiagram-99d0ae7c-34a0af5a.js → gitGraphDiagram-99d0ae7c-41e1aa3f.js} +1 -1
  69. rasa/core/channels/inspector/dist/assets/{index-2c4b9a3b-42ba3e3d.js → index-2c4b9a3b-e6f2af62.js} +1 -1
  70. rasa/core/channels/inspector/dist/assets/{index-37817b51.js → index-e793d777.js} +3 -3
  71. rasa/core/channels/inspector/dist/assets/{infoDiagram-736b4530-6b731386.js → infoDiagram-736b4530-8ceba4db.js} +1 -1
  72. rasa/core/channels/inspector/dist/assets/{journeyDiagram-df861f2b-e8579ac6.js → journeyDiagram-df861f2b-960d3809.js} +1 -1
  73. rasa/core/channels/inspector/dist/assets/{layout-89e6403a.js → layout-498807d8.js} +1 -1
  74. rasa/core/channels/inspector/dist/assets/{line-dc73d3fc.js → line-eeccc4e2.js} +1 -1
  75. rasa/core/channels/inspector/dist/assets/{linear-f5b1d2bc.js → linear-8a078617.js} +1 -1
  76. rasa/core/channels/inspector/dist/assets/{mindmap-definition-beec6740-82cb74fa.js → mindmap-definition-beec6740-396d17dd.js} +1 -1
  77. rasa/core/channels/inspector/dist/assets/{pieDiagram-dbbf0591-bdf5f29b.js → pieDiagram-dbbf0591-dc9b5e1b.js} +1 -1
  78. rasa/core/channels/inspector/dist/assets/{quadrantDiagram-4d7f4fd6-c7a0cbe4.js → quadrantDiagram-4d7f4fd6-a08cba6d.js} +1 -1
  79. rasa/core/channels/inspector/dist/assets/{requirementDiagram-6fc4c22a-7ec5410f.js → requirementDiagram-6fc4c22a-87242b9e.js} +1 -1
  80. rasa/core/channels/inspector/dist/assets/{sankeyDiagram-8f13d901-caee5554.js → sankeyDiagram-8f13d901-53f6f391.js} +1 -1
  81. rasa/core/channels/inspector/dist/assets/{sequenceDiagram-b655622a-2935f8db.js → sequenceDiagram-b655622a-715c9c20.js} +1 -1
  82. rasa/core/channels/inspector/dist/assets/{stateDiagram-59f0c015-8f5d9693.js → stateDiagram-59f0c015-2e8fb31f.js} +1 -1
  83. rasa/core/channels/inspector/dist/assets/{stateDiagram-v2-2b26beab-d565d1de.js → stateDiagram-v2-2b26beab-7e2d2aa0.js} +1 -1
  84. rasa/core/channels/inspector/dist/assets/{styles-080da4f6-75ad421d.js → styles-080da4f6-4420cea6.js} +1 -1
  85. rasa/core/channels/inspector/dist/assets/{styles-3dcbcfbf-7e764226.js → styles-3dcbcfbf-28676cf4.js} +1 -1
  86. rasa/core/channels/inspector/dist/assets/{styles-9c745c82-7a4e0e61.js → styles-9c745c82-cef936a6.js} +1 -1
  87. rasa/core/channels/inspector/dist/assets/{svgDrawCommon-4835440b-4019d1bf.js → svgDrawCommon-4835440b-151251e9.js} +1 -1
  88. rasa/core/channels/inspector/dist/assets/{timeline-definition-5b62e21b-01ea12df.js → timeline-definition-5b62e21b-0d39bdb2.js} +1 -1
  89. rasa/core/channels/inspector/dist/assets/{xychartDiagram-2b33534f-89407137.js → xychartDiagram-2b33534f-a03fa445.js} +1 -1
  90. rasa/core/channels/inspector/dist/index.html +3 -1
  91. rasa/core/channels/inspector/index.html +2 -0
  92. rasa/core/channels/inspector/src/App.tsx +1 -4
  93. rasa/core/channels/mattermost.py +4 -4
  94. rasa/core/channels/rasa_chat.py +4 -4
  95. rasa/core/channels/rest.py +11 -12
  96. rasa/core/channels/rocketchat.py +4 -3
  97. rasa/core/channels/slack.py +6 -5
  98. rasa/core/channels/socketio.py +7 -28
  99. rasa/core/channels/studio_chat.py +193 -0
  100. rasa/core/channels/telegram.py +73 -41
  101. rasa/core/channels/twilio.py +3 -3
  102. rasa/core/channels/vier_cvg.py +2 -2
  103. rasa/core/channels/voice_ready/audiocodes.py +8 -8
  104. rasa/core/channels/voice_ready/jambonz.py +5 -5
  105. rasa/core/channels/voice_ready/jambonz_protocol.py +3 -4
  106. rasa/core/channels/voice_ready/twilio_voice.py +9 -8
  107. rasa/core/channels/voice_ready/utils.py +1 -1
  108. rasa/core/channels/voice_stream/asr/asr_engine.py +2 -2
  109. rasa/core/channels/voice_stream/asr/azure.py +2 -2
  110. rasa/core/channels/voice_stream/asr/deepgram.py +57 -16
  111. rasa/core/channels/voice_stream/browser_audio.py +9 -6
  112. rasa/core/channels/voice_stream/call_state.py +2 -1
  113. rasa/core/channels/voice_stream/tts/azure.py +1 -2
  114. rasa/core/channels/voice_stream/tts/cartesia.py +5 -4
  115. rasa/core/channels/voice_stream/tts/tts_cache.py +3 -2
  116. rasa/core/channels/voice_stream/tts/tts_engine.py +1 -1
  117. rasa/core/channels/voice_stream/twilio_media_streams.py +10 -7
  118. rasa/core/channels/voice_stream/util.py +1 -1
  119. rasa/core/channels/voice_stream/voice_channel.py +15 -15
  120. rasa/core/channels/webexteams.py +3 -4
  121. rasa/core/evaluation/marker.py +7 -6
  122. rasa/core/evaluation/marker_base.py +15 -16
  123. rasa/core/evaluation/marker_stats.py +3 -4
  124. rasa/core/evaluation/marker_tracker_loader.py +5 -4
  125. rasa/core/exporter.py +4 -4
  126. rasa/core/featurizers/precomputation.py +8 -8
  127. rasa/core/featurizers/single_state_featurizer.py +7 -7
  128. rasa/core/featurizers/tracker_featurizers.py +13 -13
  129. rasa/core/http_interpreter.py +3 -4
  130. rasa/core/information_retrieval/__init__.py +1 -1
  131. rasa/core/information_retrieval/faiss.py +4 -4
  132. rasa/core/information_retrieval/information_retrieval.py +2 -2
  133. rasa/core/information_retrieval/milvus.py +3 -3
  134. rasa/core/information_retrieval/qdrant.py +3 -3
  135. rasa/core/jobs.py +1 -0
  136. rasa/core/lock.py +2 -3
  137. rasa/core/lock_store.py +3 -3
  138. rasa/core/migrate.py +11 -8
  139. rasa/core/nlg/__init__.py +1 -1
  140. rasa/core/nlg/callback.py +2 -3
  141. rasa/core/nlg/contextual_response_rephraser.py +63 -13
  142. rasa/core/nlg/generator.py +2 -2
  143. rasa/core/nlg/interpolator.py +4 -3
  144. rasa/core/nlg/response.py +3 -4
  145. rasa/core/nlg/summarize.py +1 -0
  146. rasa/core/persistor.py +3 -3
  147. rasa/core/policies/ensemble.py +10 -9
  148. rasa/core/policies/enterprise_search_policy.py +86 -21
  149. rasa/core/policies/flow_policy.py +13 -14
  150. rasa/core/policies/flows/flow_executor.py +35 -11
  151. rasa/core/policies/intentless_policy.py +6 -7
  152. rasa/core/policies/memoization.py +22 -20
  153. rasa/core/policies/policy.py +24 -22
  154. rasa/core/policies/rule_policy.py +37 -36
  155. rasa/core/policies/ted_policy.py +87 -85
  156. rasa/core/policies/unexpected_intent_policy.py +77 -75
  157. rasa/core/processor.py +96 -65
  158. rasa/core/run.py +1 -1
  159. rasa/core/secrets_manager/endpoints.py +2 -3
  160. rasa/core/secrets_manager/factory.py +2 -3
  161. rasa/core/secrets_manager/secret_manager.py +2 -3
  162. rasa/core/secrets_manager/vault.py +2 -2
  163. rasa/core/test.py +30 -30
  164. rasa/core/tracker_store.py +15 -15
  165. rasa/core/train.py +1 -1
  166. rasa/core/training/__init__.py +2 -2
  167. rasa/core/training/converters/responses_prefix_converter.py +1 -2
  168. rasa/core/training/interactive.py +13 -13
  169. rasa/core/training/story_conflict.py +4 -5
  170. rasa/core/training/training.py +3 -5
  171. rasa/core/utils.py +5 -5
  172. rasa/core/visualize.py +1 -1
  173. rasa/dialogue_understanding/coexistence/intent_based_router.py +2 -2
  174. rasa/dialogue_understanding/coexistence/llm_based_router.py +5 -5
  175. rasa/dialogue_understanding/commands/__init__.py +22 -22
  176. rasa/dialogue_understanding/commands/can_not_handle_command.py +20 -1
  177. rasa/dialogue_understanding/commands/cancel_flow_command.py +18 -7
  178. rasa/dialogue_understanding/commands/change_flow_command.py +18 -2
  179. rasa/dialogue_understanding/commands/chit_chat_answer_command.py +18 -4
  180. rasa/dialogue_understanding/commands/clarify_command.py +19 -3
  181. rasa/dialogue_understanding/commands/command.py +19 -3
  182. rasa/dialogue_understanding/commands/correct_slots_command.py +3 -3
  183. rasa/dialogue_understanding/commands/error_command.py +1 -1
  184. rasa/dialogue_understanding/commands/free_form_answer_command.py +2 -1
  185. rasa/dialogue_understanding/commands/handle_code_change_command.py +2 -2
  186. rasa/dialogue_understanding/commands/human_handoff_command.py +16 -4
  187. rasa/dialogue_understanding/commands/knowledge_answer_command.py +18 -4
  188. rasa/dialogue_understanding/commands/noop_command.py +2 -1
  189. rasa/dialogue_understanding/commands/repeat_bot_messages_command.py +16 -4
  190. rasa/dialogue_understanding/commands/restart_command.py +2 -5
  191. rasa/dialogue_understanding/commands/session_end_command.py +3 -5
  192. rasa/dialogue_understanding/commands/session_start_command.py +3 -5
  193. rasa/dialogue_understanding/commands/set_slot_command.py +27 -3
  194. rasa/dialogue_understanding/commands/skip_question_command.py +16 -4
  195. rasa/dialogue_understanding/commands/start_flow_command.py +17 -2
  196. rasa/dialogue_understanding/commands/user_silence_command.py +3 -5
  197. rasa/dialogue_understanding/commands/utils.py +64 -45
  198. rasa/dialogue_understanding/constants.py +1 -0
  199. rasa/dialogue_understanding/generator/command_generator.py +2 -119
  200. rasa/dialogue_understanding/generator/command_parser.py +201 -0
  201. rasa/dialogue_understanding/generator/constants.py +2 -2
  202. rasa/dialogue_understanding/generator/flow_retrieval.py +7 -7
  203. rasa/dialogue_understanding/generator/llm_based_command_generator.py +11 -57
  204. rasa/dialogue_understanding/generator/llm_command_generator.py +2 -1
  205. rasa/dialogue_understanding/generator/multi_step/multi_step_llm_command_generator.py +67 -120
  206. rasa/dialogue_understanding/generator/nlu_command_adapter.py +8 -10
  207. rasa/dialogue_understanding/generator/single_step/single_step_llm_command_generator.py +32 -93
  208. rasa/dialogue_understanding/generator/utils.py +45 -0
  209. rasa/dialogue_understanding/patterns/cancel.py +2 -1
  210. rasa/dialogue_understanding/patterns/cannot_handle.py +1 -0
  211. rasa/dialogue_understanding/patterns/chitchat.py +1 -1
  212. rasa/dialogue_understanding/patterns/clarify.py +2 -1
  213. rasa/dialogue_understanding/patterns/code_change.py +2 -0
  214. rasa/dialogue_understanding/patterns/collect_information.py +6 -3
  215. rasa/dialogue_understanding/patterns/completed.py +1 -1
  216. rasa/dialogue_understanding/patterns/continue_interrupted.py +10 -1
  217. rasa/dialogue_understanding/patterns/correction.py +4 -2
  218. rasa/dialogue_understanding/patterns/human_handoff.py +1 -1
  219. rasa/dialogue_understanding/patterns/internal_error.py +1 -0
  220. rasa/dialogue_understanding/patterns/search.py +1 -1
  221. rasa/dialogue_understanding/patterns/session_start.py +1 -1
  222. rasa/dialogue_understanding/patterns/skip_question.py +1 -0
  223. rasa/dialogue_understanding/patterns/user_silence.py +1 -1
  224. rasa/dialogue_understanding/processor/command_processor.py +13 -9
  225. rasa/dialogue_understanding/processor/command_processor_component.py +1 -1
  226. rasa/dialogue_understanding/stack/dialogue_stack.py +4 -3
  227. rasa/dialogue_understanding/stack/frames/__init__.py +2 -2
  228. rasa/dialogue_understanding/stack/frames/chit_chat_frame.py +4 -1
  229. rasa/dialogue_understanding/stack/frames/dialogue_stack_frame.py +2 -3
  230. rasa/dialogue_understanding/stack/frames/flow_stack_frame.py +5 -2
  231. rasa/dialogue_understanding/stack/frames/search_frame.py +4 -1
  232. rasa/dialogue_understanding/stack/utils.py +8 -4
  233. rasa/dialogue_understanding/utils.py +121 -2
  234. rasa/dialogue_understanding_test/README.md +379 -0
  235. rasa/dialogue_understanding_test/command_comparison.py +60 -0
  236. rasa/dialogue_understanding_test/command_metric_calculation.py +110 -4
  237. rasa/dialogue_understanding_test/constants.py +6 -1
  238. rasa/dialogue_understanding_test/du_test_case.py +252 -38
  239. rasa/dialogue_understanding_test/du_test_result.py +291 -2
  240. rasa/dialogue_understanding_test/du_test_runner.py +239 -10
  241. rasa/dialogue_understanding_test/du_test_schema.yml +161 -0
  242. rasa/dialogue_understanding_test/io.py +338 -20
  243. rasa/dialogue_understanding_test/test_case_simulation/__init__.py +0 -0
  244. rasa/dialogue_understanding_test/test_case_simulation/exception.py +28 -0
  245. rasa/dialogue_understanding_test/test_case_simulation/test_case_tracker_simulator.py +336 -0
  246. rasa/dialogue_understanding_test/utils.py +70 -0
  247. rasa/dialogue_understanding_test/validation.py +59 -4
  248. rasa/e2e_test/aggregate_test_stats_calculator.py +1 -1
  249. rasa/e2e_test/assertions.py +1 -1
  250. rasa/e2e_test/e2e_config.py +1 -1
  251. rasa/e2e_test/e2e_test_case.py +3 -4
  252. rasa/e2e_test/e2e_test_converter.py +2 -3
  253. rasa/e2e_test/e2e_test_coverage_report.py +6 -6
  254. rasa/e2e_test/e2e_test_result.py +1 -1
  255. rasa/e2e_test/e2e_test_runner.py +134 -31
  256. rasa/e2e_test/stub_custom_action.py +1 -1
  257. rasa/e2e_test/utils/e2e_yaml_utils.py +1 -1
  258. rasa/e2e_test/utils/io.py +132 -65
  259. rasa/e2e_test/utils/validation.py +1 -1
  260. rasa/engine/caching.py +5 -7
  261. rasa/engine/constants.py +1 -1
  262. rasa/engine/graph.py +2 -2
  263. rasa/engine/recipes/default_components.py +13 -15
  264. rasa/engine/recipes/recipe.py +2 -2
  265. rasa/engine/runner/dask.py +2 -2
  266. rasa/engine/runner/interface.py +1 -0
  267. rasa/engine/storage/local_model_storage.py +5 -4
  268. rasa/engine/storage/resource.py +2 -1
  269. rasa/engine/storage/storage.py +5 -3
  270. rasa/engine/training/components.py +2 -1
  271. rasa/engine/training/fingerprinting.py +4 -2
  272. rasa/engine/training/graph_trainer.py +4 -4
  273. rasa/engine/training/hooks.py +2 -2
  274. rasa/engine/validation.py +34 -33
  275. rasa/exceptions.py +3 -2
  276. rasa/graph_components/converters/nlu_message_converter.py +3 -3
  277. rasa/graph_components/providers/domain_for_core_training_provider.py +3 -3
  278. rasa/graph_components/providers/domain_provider.py +3 -2
  279. rasa/graph_components/providers/flows_provider.py +2 -3
  280. rasa/graph_components/providers/forms_provider.py +4 -4
  281. rasa/graph_components/providers/nlu_training_data_provider.py +5 -3
  282. rasa/graph_components/providers/responses_provider.py +4 -4
  283. rasa/graph_components/providers/rule_only_provider.py +3 -2
  284. rasa/graph_components/providers/story_graph_provider.py +8 -8
  285. rasa/graph_components/providers/training_tracker_provider.py +3 -2
  286. rasa/graph_components/validators/default_recipe_validator.py +16 -16
  287. rasa/graph_components/validators/finetuning_validator.py +10 -8
  288. rasa/hooks.py +18 -12
  289. rasa/jupyter.py +2 -2
  290. rasa/llm_fine_tuning/annotation_module.py +4 -4
  291. rasa/llm_fine_tuning/conversations.py +6 -6
  292. rasa/llm_fine_tuning/llm_data_preparation_module.py +1 -1
  293. rasa/llm_fine_tuning/paraphrasing/conversation_rephraser.py +4 -4
  294. rasa/llm_fine_tuning/paraphrasing/rephrase_validator.py +1 -1
  295. rasa/llm_fine_tuning/paraphrasing_module.py +1 -1
  296. rasa/llm_fine_tuning/storage.py +3 -3
  297. rasa/markers/marker.py +2 -3
  298. rasa/markers/marker_base.py +1 -2
  299. rasa/markers/upload.py +2 -2
  300. rasa/markers/validate.py +2 -3
  301. rasa/model.py +3 -5
  302. rasa/model_manager/config.py +1 -1
  303. rasa/model_manager/model_api.py +5 -4
  304. rasa/model_manager/runner_service.py +6 -6
  305. rasa/model_manager/socket_bridge.py +8 -3
  306. rasa/model_manager/studio_jwt_auth.py +1 -0
  307. rasa/model_manager/trainer_service.py +9 -7
  308. rasa/model_manager/utils.py +1 -1
  309. rasa/model_manager/warm_rasa_process.py +14 -8
  310. rasa/model_service.py +5 -6
  311. rasa/model_testing.py +13 -15
  312. rasa/nlu/classifiers/diet_classifier.py +72 -73
  313. rasa/nlu/classifiers/fallback_classifier.py +9 -11
  314. rasa/nlu/classifiers/keyword_intent_classifier.py +7 -6
  315. rasa/nlu/classifiers/logistic_regression_classifier.py +3 -3
  316. rasa/nlu/classifiers/mitie_intent_classifier.py +5 -4
  317. rasa/nlu/classifiers/regex_message_handler.py +3 -2
  318. rasa/nlu/classifiers/sklearn_intent_classifier.py +2 -2
  319. rasa/nlu/convert.py +2 -2
  320. rasa/nlu/emulators/dialogflow.py +3 -3
  321. rasa/nlu/emulators/luis.py +5 -5
  322. rasa/nlu/emulators/no_emulator.py +1 -0
  323. rasa/nlu/emulators/wit.py +4 -4
  324. rasa/nlu/extractors/crf_entity_extractor.py +11 -11
  325. rasa/nlu/extractors/duckling_entity_extractor.py +7 -6
  326. rasa/nlu/extractors/entity_synonyms.py +10 -9
  327. rasa/nlu/extractors/extractor.py +16 -16
  328. rasa/nlu/extractors/mitie_entity_extractor.py +10 -9
  329. rasa/nlu/extractors/regex_entity_extractor.py +11 -10
  330. rasa/nlu/extractors/spacy_entity_extractor.py +2 -2
  331. rasa/nlu/featurizers/dense_featurizer/convert_featurizer.py +15 -14
  332. rasa/nlu/featurizers/dense_featurizer/dense_featurizer.py +2 -1
  333. rasa/nlu/featurizers/dense_featurizer/lm_featurizer.py +10 -9
  334. rasa/nlu/featurizers/dense_featurizer/mitie_featurizer.py +9 -7
  335. rasa/nlu/featurizers/dense_featurizer/spacy_featurizer.py +13 -12
  336. rasa/nlu/featurizers/featurizer.py +5 -4
  337. rasa/nlu/featurizers/sparse_featurizer/count_vectors_featurizer.py +6 -6
  338. rasa/nlu/featurizers/sparse_featurizer/lexical_syntactic_featurizer.py +4 -4
  339. rasa/nlu/featurizers/sparse_featurizer/regex_featurizer.py +4 -4
  340. rasa/nlu/featurizers/sparse_featurizer/sparse_featurizer.py +2 -0
  341. rasa/nlu/model.py +0 -1
  342. rasa/nlu/selectors/response_selector.py +67 -68
  343. rasa/nlu/test.py +38 -38
  344. rasa/nlu/tokenizers/jieba_tokenizer.py +1 -2
  345. rasa/nlu/tokenizers/mitie_tokenizer.py +2 -2
  346. rasa/nlu/tokenizers/spacy_tokenizer.py +3 -3
  347. rasa/nlu/tokenizers/tokenizer.py +6 -7
  348. rasa/nlu/tokenizers/whitespace_tokenizer.py +1 -1
  349. rasa/nlu/utils/bilou_utils.py +7 -7
  350. rasa/nlu/utils/hugging_face/registry.py +22 -22
  351. rasa/nlu/utils/hugging_face/transformers_pre_post_processors.py +2 -1
  352. rasa/nlu/utils/mitie_utils.py +2 -1
  353. rasa/nlu/utils/pattern_utils.py +1 -1
  354. rasa/nlu/utils/spacy_utils.py +3 -3
  355. rasa/plugin.py +12 -1
  356. rasa/server.py +37 -1
  357. rasa/shared/constants.py +22 -2
  358. rasa/shared/core/command_payload_reader.py +15 -7
  359. rasa/shared/core/constants.py +4 -1
  360. rasa/shared/core/conversation.py +1 -2
  361. rasa/shared/core/events.py +47 -37
  362. rasa/shared/core/flows/__init__.py +0 -1
  363. rasa/shared/core/flows/flow.py +11 -11
  364. rasa/shared/core/flows/flow_step.py +19 -13
  365. rasa/shared/core/flows/flow_step_links.py +21 -14
  366. rasa/shared/core/flows/flow_step_sequence.py +6 -4
  367. rasa/shared/core/flows/flows_list.py +3 -3
  368. rasa/shared/core/flows/nlu_trigger.py +1 -1
  369. rasa/shared/core/flows/steps/__init__.py +2 -2
  370. rasa/shared/core/flows/steps/action.py +4 -3
  371. rasa/shared/core/flows/steps/call.py +4 -4
  372. rasa/shared/core/flows/steps/collect.py +7 -4
  373. rasa/shared/core/flows/steps/continuation.py +3 -1
  374. rasa/shared/core/flows/steps/end.py +3 -1
  375. rasa/shared/core/flows/steps/internal.py +3 -2
  376. rasa/shared/core/flows/steps/link.py +6 -4
  377. rasa/shared/core/flows/steps/no_operation.py +7 -5
  378. rasa/shared/core/flows/steps/set_slots.py +4 -3
  379. rasa/shared/core/flows/steps/start.py +3 -1
  380. rasa/shared/core/flows/utils.py +1 -0
  381. rasa/shared/core/flows/validation.py +3 -5
  382. rasa/shared/core/generator.py +20 -21
  383. rasa/shared/core/slot_mappings.py +15 -15
  384. rasa/shared/core/slots.py +3 -3
  385. rasa/shared/core/trackers.py +31 -31
  386. rasa/shared/core/training_data/loading.py +1 -1
  387. rasa/shared/core/training_data/story_reader/story_reader.py +3 -3
  388. rasa/shared/core/training_data/story_reader/story_step_builder.py +4 -4
  389. rasa/shared/core/training_data/story_reader/yaml_story_reader.py +29 -31
  390. rasa/shared/core/training_data/story_writer/yaml_story_writer.py +22 -24
  391. rasa/shared/core/training_data/structures.py +11 -12
  392. rasa/shared/core/training_data/visualization.py +10 -10
  393. rasa/shared/data.py +6 -6
  394. rasa/shared/engine/caching.py +0 -1
  395. rasa/shared/exceptions.py +2 -2
  396. rasa/shared/importers/rasa.py +5 -6
  397. rasa/shared/importers/utils.py +1 -1
  398. rasa/shared/nlu/constants.py +3 -0
  399. rasa/shared/nlu/training_data/entities_parser.py +6 -6
  400. rasa/shared/nlu/training_data/features.py +3 -3
  401. rasa/shared/nlu/training_data/formats/__init__.py +1 -1
  402. rasa/shared/nlu/training_data/formats/dialogflow.py +4 -5
  403. rasa/shared/nlu/training_data/formats/luis.py +7 -8
  404. rasa/shared/nlu/training_data/formats/rasa.py +4 -5
  405. rasa/shared/nlu/training_data/formats/rasa_yaml.py +17 -16
  406. rasa/shared/nlu/training_data/formats/readerwriter.py +8 -11
  407. rasa/shared/nlu/training_data/formats/wit.py +3 -4
  408. rasa/shared/nlu/training_data/loading.py +4 -4
  409. rasa/shared/nlu/training_data/lookup_tables_parser.py +1 -1
  410. rasa/shared/nlu/training_data/message.py +13 -14
  411. rasa/shared/nlu/training_data/schemas/data_schema.py +1 -1
  412. rasa/shared/nlu/training_data/schemas/responses.yml +1 -0
  413. rasa/shared/nlu/training_data/synonyms_parser.py +3 -3
  414. rasa/shared/nlu/training_data/training_data.py +12 -13
  415. rasa/shared/nlu/training_data/util.py +11 -10
  416. rasa/shared/providers/_configs/azure_openai_client_config.py +3 -119
  417. rasa/shared/providers/_configs/client_config.py +1 -3
  418. rasa/shared/providers/_configs/default_litellm_client_config.py +1 -3
  419. rasa/shared/providers/_configs/huggingface_local_embedding_client_config.py +1 -3
  420. rasa/shared/providers/_configs/litellm_router_client_config.py +1 -3
  421. rasa/shared/providers/_configs/model_group_config.py +2 -7
  422. rasa/shared/providers/_configs/openai_client_config.py +1 -3
  423. rasa/shared/providers/_configs/rasa_llm_client_config.py +1 -3
  424. rasa/shared/providers/_configs/self_hosted_llm_client_config.py +1 -3
  425. rasa/shared/providers/_configs/utils.py +1 -0
  426. rasa/shared/providers/_ssl_verification_utils.py +5 -6
  427. rasa/shared/providers/_utils.py +5 -5
  428. rasa/shared/providers/embedding/_base_litellm_embedding_client.py +1 -1
  429. rasa/shared/providers/embedding/azure_openai_embedding_client.py +3 -26
  430. rasa/shared/providers/embedding/embedding_client.py +1 -1
  431. rasa/shared/providers/embedding/litellm_router_embedding_client.py +1 -3
  432. rasa/shared/providers/llm/_base_litellm_client.py +1 -3
  433. rasa/shared/providers/llm/azure_openai_llm_client.py +25 -79
  434. rasa/shared/providers/llm/default_litellm_llm_client.py +1 -3
  435. rasa/shared/providers/llm/litellm_router_llm_client.py +2 -21
  436. rasa/shared/providers/llm/llm_client.py +1 -3
  437. rasa/shared/providers/llm/llm_response.py +41 -2
  438. rasa/shared/providers/llm/openai_llm_client.py +3 -9
  439. rasa/shared/providers/llm/rasa_llm_client.py +6 -20
  440. rasa/shared/providers/llm/self_hosted_llm_client.py +3 -9
  441. rasa/shared/providers/mappings.py +18 -19
  442. rasa/shared/providers/router/_base_litellm_router_client.py +1 -3
  443. rasa/shared/providers/router/router_client.py +1 -3
  444. rasa/shared/utils/cli.py +1 -1
  445. rasa/shared/utils/common.py +1 -1
  446. rasa/shared/utils/health_check/embeddings_health_check_mixin.py +1 -1
  447. rasa/shared/utils/health_check/health_check.py +3 -3
  448. rasa/shared/utils/health_check/llm_health_check_mixin.py +1 -1
  449. rasa/shared/utils/io.py +1 -1
  450. rasa/shared/utils/llm.py +5 -9
  451. rasa/shared/utils/pykwalify_extensions.py +1 -1
  452. rasa/shared/utils/schemas/events.py +1 -1
  453. rasa/shared/utils/yaml.py +29 -26
  454. rasa/studio/auth.py +3 -3
  455. rasa/studio/config.py +1 -2
  456. rasa/studio/data_handler.py +3 -3
  457. rasa/studio/download.py +1 -1
  458. rasa/studio/results_logger.py +3 -3
  459. rasa/studio/upload.py +2 -0
  460. rasa/telemetry.py +31 -0
  461. rasa/tracing/config.py +3 -3
  462. rasa/tracing/instrumentation/attribute_extractors.py +54 -0
  463. rasa/tracing/instrumentation/instrumentation.py +25 -5
  464. rasa/tracing/instrumentation/intentless_policy_instrumentation.py +1 -1
  465. rasa/tracing/instrumentation/metrics.py +11 -11
  466. rasa/tracing/metric_instrument_provider.py +14 -14
  467. rasa/utils/common.py +12 -9
  468. rasa/utils/endpoints.py +1 -1
  469. rasa/utils/io.py +7 -7
  470. rasa/utils/licensing.py +3 -4
  471. rasa/utils/log_utils.py +7 -6
  472. rasa/utils/ml_utils.py +1 -0
  473. rasa/utils/plotting.py +3 -3
  474. rasa/utils/sanic_error_handler.py +1 -1
  475. rasa/utils/tensorflow/callback.py +2 -2
  476. rasa/utils/tensorflow/crf.py +2 -2
  477. rasa/utils/tensorflow/data_generator.py +5 -5
  478. rasa/utils/tensorflow/environment.py +3 -3
  479. rasa/utils/tensorflow/feature_array.py +2 -3
  480. rasa/utils/tensorflow/layers.py +18 -12
  481. rasa/utils/tensorflow/layers_utils.py +2 -1
  482. rasa/utils/tensorflow/metrics.py +2 -2
  483. rasa/utils/tensorflow/model_data.py +7 -7
  484. rasa/utils/tensorflow/model_data_utils.py +10 -9
  485. rasa/utils/tensorflow/models.py +31 -32
  486. rasa/utils/tensorflow/rasa_layers.py +20 -19
  487. rasa/utils/tensorflow/types.py +2 -1
  488. rasa/utils/train_utils.py +23 -21
  489. rasa/utils/url_tools.py +1 -1
  490. rasa/validator.py +17 -1
  491. rasa/version.py +1 -1
  492. {rasa_pro-3.12.0.dev2.dist-info → rasa_pro-3.12.0.dev4.dist-info}/METADATA +9 -10
  493. rasa_pro-3.12.0.dev4.dist-info/RECORD +800 -0
  494. rasa/core/channels/inspector/dist/assets/flowDiagram-v2-855bc5b3-b080d6f2.js +0 -1
  495. rasa/shared/providers/_configs/azure_entra_id_client_creds.py +0 -40
  496. rasa/shared/providers/_configs/azure_entra_id_config.py +0 -533
  497. rasa/shared/providers/_configs/oauth_config.py +0 -33
  498. rasa/shared/providers/constants.py +0 -6
  499. rasa_pro-3.12.0.dev2.dist-info/RECORD +0 -793
  500. {rasa_pro-3.12.0.dev2.dist-info → rasa_pro-3.12.0.dev4.dist-info}/NOTICE +0 -0
  501. {rasa_pro-3.12.0.dev2.dist-info → rasa_pro-3.12.0.dev4.dist-info}/WHEEL +0 -0
  502. {rasa_pro-3.12.0.dev2.dist-info → rasa_pro-3.12.0.dev4.dist-info}/entry_points.txt +0 -0
@@ -1,17 +1,20 @@
1
- from typing import List, Optional, Set, Tuple
2
1
  import typing
2
+ from typing import List, Optional, Set, Tuple
3
+
3
4
  from rasa.dialogue_understanding.patterns.collect_information import (
4
5
  CollectInformationPatternFlowStackFrame,
5
6
  )
6
- from rasa.dialogue_understanding.stack.frames import BaseFlowStackFrame
7
7
  from rasa.dialogue_understanding.stack.dialogue_stack import DialogueStack
8
- from rasa.dialogue_understanding.stack.frames import UserFlowStackFrame
8
+ from rasa.dialogue_understanding.stack.frames import (
9
+ BaseFlowStackFrame,
10
+ UserFlowStackFrame,
11
+ )
9
12
  from rasa.dialogue_understanding.stack.frames.flow_stack_frame import FlowStackFrameType
10
13
  from rasa.dialogue_understanding.stack.frames.pattern_frame import PatternFlowStackFrame
14
+ from rasa.shared.core.flows import FlowsList
11
15
  from rasa.shared.core.flows.steps.collect import CollectInformationFlowStep
12
16
  from rasa.shared.core.flows.steps.constants import END_STEP
13
17
  from rasa.shared.core.flows.steps.continuation import ContinueFlowStep
14
- from rasa.shared.core.flows import FlowsList
15
18
 
16
19
  if typing.TYPE_CHECKING:
17
20
  from rasa.shared.core.trackers import DialogueStateTracker
@@ -77,6 +80,7 @@ def top_user_flow_frame(dialogue_stack: DialogueStack) -> Optional[UserFlowStack
77
80
  if (
78
81
  isinstance(frame, UserFlowStackFrame)
79
82
  and frame.frame_type != FlowStackFrameType.CALL
83
+ and frame.frame_type != FlowStackFrameType.LINK
80
84
  ):
81
85
  return frame
82
86
  return None
@@ -1,7 +1,26 @@
1
1
  from contextlib import contextmanager
2
- from typing import Generator
2
+ from typing import Any, Dict, Generator, List, Optional, Text
3
3
 
4
- record_commands_and_prompts = False
4
+ from rasa.dialogue_understanding.commands import Command
5
+ from rasa.dialogue_understanding.constants import (
6
+ RASA_RECORD_COMMANDS_AND_PROMPTS_ENV_VAR_NAME,
7
+ )
8
+ from rasa.shared.nlu.constants import (
9
+ KEY_COMPONENT_NAME,
10
+ KEY_LLM_RESPONSE_METADATA,
11
+ KEY_PROMPT_NAME,
12
+ KEY_SYSTEM_PROMPT,
13
+ KEY_USER_PROMPT,
14
+ PREDICTED_COMMANDS,
15
+ PROMPTS,
16
+ )
17
+ from rasa.shared.nlu.training_data.message import Message
18
+ from rasa.shared.providers.llm.llm_response import LLMResponse
19
+ from rasa.utils.common import get_bool_env_variable
20
+
21
+ record_commands_and_prompts = get_bool_env_variable(
22
+ RASA_RECORD_COMMANDS_AND_PROMPTS_ENV_VAR_NAME, False
23
+ )
5
24
 
6
25
 
7
26
  @contextmanager
@@ -12,3 +31,103 @@ def set_record_commands_and_prompts() -> Generator:
12
31
  yield
13
32
  finally:
14
33
  record_commands_and_prompts = False
34
+
35
+
36
+ def add_commands_to_message_parse_data(
37
+ message: Message, component_name: str, commands: List[Command]
38
+ ) -> None:
39
+ """Add commands to the message parse data.
40
+
41
+ Commands are only added in case the flag 'record_commands_and_prompts' is set.
42
+ Example of predicted commands in the message parse data:
43
+ Message(data={
44
+ PREDICTED_COMMANDS: {
45
+ "MultiStepLLMCommandGenerator": [
46
+ {"command": "set_slot", "name": "slot_name", "value": "slot_value"},
47
+ ],
48
+ "NLUCommandAdapter": [
49
+ {"command": "start_flow", "name": "test_flow"},
50
+ ]
51
+ }
52
+ })
53
+ """
54
+ # only set commands if the flag "record_commands_and_prompts" is set to True
55
+ if not record_commands_and_prompts:
56
+ return
57
+
58
+ commands_as_dict = [command.as_dict() for command in commands]
59
+
60
+ if message.get(PREDICTED_COMMANDS) is not None:
61
+ predicted_commands = message.get(PREDICTED_COMMANDS)
62
+ if component_name in predicted_commands:
63
+ predicted_commands[component_name].extend(commands_as_dict)
64
+ else:
65
+ predicted_commands[component_name] = commands_as_dict
66
+ else:
67
+ predicted_commands = {component_name: commands_as_dict}
68
+
69
+ message.set(
70
+ PREDICTED_COMMANDS,
71
+ predicted_commands,
72
+ add_to_output=True,
73
+ )
74
+
75
+
76
+ def add_prompt_to_message_parse_data(
77
+ message: Message,
78
+ component_name: str,
79
+ prompt_name: str,
80
+ user_prompt: str,
81
+ system_prompt: Optional[str] = None,
82
+ llm_response: Optional[LLMResponse] = None,
83
+ ) -> None:
84
+ """Add prompt to the message parse data.
85
+
86
+ Prompt is only added in case the flag 'record_commands_and_prompts' is set.
87
+ Example of prompts in the message parse data:
88
+ Message(data={
89
+ PROMPTS: [
90
+ {
91
+ "component_name": "MultiStepLLMCommandGenerator",
92
+ "prompt_name": "fill_slots_prompt",
93
+ "user_prompt": "...",
94
+ "system_prompt": "...",
95
+ "llm_response_metadata": { ... }
96
+ },
97
+ {
98
+ "component_name": "MultiStepLLMCommandGenerator",
99
+ "prompt_name": "handle_flows_prompt",
100
+ "user_prompt": "...",
101
+ "system_prompt": "...",
102
+ "llm_response_metadata": { ... }
103
+ },
104
+ {
105
+ "component_name": "SingleStepLLMCommandGenerator",
106
+ "prompt_name": "prompt_template",
107
+ "user_prompt": "...",
108
+ "system_prompt": "...",
109
+ "llm_response_metadata": { ... }
110
+ }
111
+ ]
112
+ }
113
+ )
114
+ """
115
+ # Only set prompt if the flag "record_commands_and_prompts" is set to True.
116
+ if not record_commands_and_prompts:
117
+ return
118
+
119
+ # Construct the dictionary with prompt details.
120
+ prompt_data: Dict[Text, Any] = {
121
+ KEY_COMPONENT_NAME: component_name,
122
+ KEY_PROMPT_NAME: prompt_name,
123
+ KEY_USER_PROMPT: user_prompt,
124
+ KEY_LLM_RESPONSE_METADATA: llm_response.to_dict() if llm_response else None,
125
+ **({KEY_SYSTEM_PROMPT: system_prompt} if system_prompt else {}),
126
+ }
127
+
128
+ # Get or create a top-level "prompts" list.
129
+ prompts = message.get(PROMPTS) or []
130
+ prompts.append(prompt_data)
131
+
132
+ # Update the message with the new prompts list.
133
+ message.set(PROMPTS, prompts, add_to_output=True)
@@ -0,0 +1,379 @@
1
+ # Dialogue Understanding Tests
2
+
3
+ Dialogue Understanding Tests (DUT) are designed to evaluate the command prediction accuracy of a
4
+ chatbot's [dialogue understanding](https://rasa.com/docs/rasa-pro/concepts/dialogue-understanding) module.
5
+ Rather than merely assessing whether a chatbot behaves as expected in
6
+ [end-to-end (E2E) tests](https://rasa.com/docs/rasa-pro/production/testing-your-assistant#end-to-end-testing), these
7
+ tests delve deeper into understanding *why* a chatbot may not be performing as anticipated.
8
+ They aim to identify discrepancies between the expected and predicted
9
+ [commands](https://rasa.com/docs/rasa-pro/concepts/dialogue-understanding#command-reference) during a conversation,
10
+ providing insights into potential pitfalls in the
11
+ [command generator](https://rasa.com/docs/rasa-pro/concepts/dialogue-understanding#commandgenerator)'s operation.
12
+
13
+ The primary focus of Dialogue Understanding Tests is the command generator,
14
+ a core component responsible for interpreting user input and orchestrating the chatbot's subsequent actions.
15
+ When updates are made to the command generator — such as switching to a different LLM or tweaking the prompt —
16
+ Dialogue Understanding Tests offer a structured approach to evaluate how accurately these changes affect command
17
+ predictions.
18
+
19
+ *Note*: Dialogue Understanding Tests only work for [CALM-based assistants](https://rasa.com/docs/rasa-pro/calm)!
20
+
21
+ ## How Dialogue Understanding Tests work
22
+
23
+ Dialogue Understanding Tests are designed to evaluate a chatbot's dialogue understanding capabilities
24
+ within a given conversational context.
25
+ In order to run Dialogue Understanding Tests, you first need to write test cases.
26
+ Each test case consists of a sequence of interactions that simulate conversations with the chatbot.
27
+ These interactions are broken down into user inputs, expected commands, and bot responses.
28
+ Each step of a test case is evaluated independently.
29
+ The predicted commands of the dialogue understanding module are compared with the expected commands
30
+ defined in the test case.
31
+ The Dialogue Understanding Test framework is able to evaluate each step of a test case, also if
32
+ a previous test step failed.
33
+ This allows for a more detailed analysis of the chatbot's performance.
34
+ After all test cases have been executed, a detailed report is generated, including metrics such as accuracy,
35
+ precision, recall, and f1-score for all commands.
36
+
37
+ ## Running Dialogue Understanding Tests
38
+
39
+ Dialogue Understanding Tests are hidden behind a feature flag. To enable the feature, set the
40
+ environment variable `RASA_PRO_BETA_DIALOGUE_UNDERSTANDING_TEST` to `true`.
41
+
42
+ To run Dialogue Understanding Tests, execute the following command:
43
+
44
+ ```bash
45
+ rasa test du <path-to-test-cases>
46
+ ```
47
+
48
+ By default, the test cases are expected to be located in the `dialogue_understanding_tests` directory.
49
+ Execute `rasa test du --help` to see the available options for running Dialogue Understanding Tests.
50
+
51
+ In order to execute any custom action that is needed by the Dialogue Understanding Tests, you need to
52
+ either start the action server in the background before running the tests via `rasa run actions` or use
53
+ [Stubbing Custom Actions](https://rasa.com/docs/rasa-pro/production/testing-your-assistant#stubbing-custom-actions).
54
+
55
+ ## Defining a Dialogue Understanding Test Case
56
+
57
+ A test case is structured as a sequence of interactions between a user and the chatbot, specifying both inputs and
58
+ expected outcomes. Each test case is composed of multiple steps, and each step consists of:
59
+
60
+ - **User Utterance**: The input message from the user.
61
+ - **Commands**: All expected commands that are generated in response to the user's message.
62
+ - **Bot Response(s)**: All expected responses from the bot, which can be either a direct textual response or a reference
63
+ to
64
+ a predefined bot response template.
65
+
66
+ Here is a sample test case:
67
+
68
+ ```yaml
69
+ test_cases:
70
+ - test_case: user_adds_contact_to_their_list
71
+ steps:
72
+ - user: I want to add someone to my contact list
73
+ commands:
74
+ - StartFlow(add_contact)
75
+ - utter: utter_ask_add_contact_handle
76
+ - user: it's @barts
77
+ commands:
78
+ - SetSlot(handle, @barts)
79
+ - bot: "What is the name of the contact you want to add?"
80
+ - user: just Bart
81
+ commands:
82
+ - SetSlot(name, Bart)
83
+ ```
84
+
85
+ *Note*: The list of commands and the list of bot responses need to be complete!
86
+
87
+ [Fixtures](https://rasa.com/docs/rasa-pro/production/testing-your-assistant#fixtures-for-pre-filled-slots),
88
+ [Metadata](https://rasa.com/docs/rasa-pro/production/testing-your-assistant#metadata-on-user-messages),
89
+ and [Stubbing Custom Actions](https://rasa.com/docs/rasa-pro/production/testing-your-assistant#stubbing-custom-actions)
90
+ known from end-to-end (E2E) tests are supported as well.
91
+ They behave the same as in E2E tests.
92
+
93
+ ### Explanation of the Commands
94
+
95
+ [Commands](https://rasa.com/docs/rasa-pro/concepts/dialogue-understanding#command-reference)
96
+ are the core components that direct the chatbot's actions in response to a user's input.
97
+ Each command represents an operation or decision the bot should make.
98
+ Here are the default commands that can be used in a test case:
99
+
100
+ - **StartFlow(flow_name)**: Starts a new flow with the specified flow name.
101
+ - **CancelFlow()**: Cancels the current flow.
102
+ - **SetSlot(slot_name, slot_value)**: Assigns a value to a specific slot.
103
+ - **Clarify(options)**: Seeks clarification by presenting options to the user.
104
+ The options are optional in Dialogue Understanding Tests.
105
+ - **ChitChat()**: Initiates a casual conversation or response.
106
+ - **SearchAndReply()**: Performs a search operation and generates a reply.
107
+ - **HumanHandoff()**: Transfers the conversation to a human agent.
108
+ - **SkipQuestion()**: The user asked to skip a certain step.
109
+ - **RepeatLastBotMessages()**: Repeats the last bot message(s) to the user.
110
+
111
+ The syntax of the commands matches the domain specific language (DSL) used in the prompt templates of the
112
+ command generators.
113
+
114
+ *Note*: It is also possible to use custom commands in the test cases
115
+ (see section "Evaluation of Custom Commands").
116
+
117
+ ### Explanation of `placeholder_generated_answer`
118
+
119
+ The `placeholder_generated_answer` is used in scenarios where a bot response is dynamically generated, such as
120
+ when the bot retrieves information from an external knowledge base.
121
+ In such cases, you may not know the exact wording of the bot's response ahead of time.
122
+ This placeholder should be used in the test case where a specific bot response is expected but may vary due
123
+ to external dynamic content or search results.
124
+ It signals that the exact bot utterance is not fixed, yet the test case recognizes and accepts a dynamically
125
+ generated response in its place.
126
+
127
+ Here is an example test case that uses `placeholder_generated_answer`:
128
+
129
+ ```yaml
130
+ test_cases:
131
+ - test_case: user asks a knowledge question during flow
132
+ steps:
133
+ - user: I want to send some money to Tre
134
+ commands:
135
+ - StartFlow(transfer_money)
136
+ - SetSlot(transfer_money_recipient, Tre)
137
+ - utter: utter_ask_transfer_money_amount_of_money
138
+ - user: btw, are these transfers free of charge?
139
+ commands:
140
+ - SearchAndReply()
141
+ - utter: placeholder_generated_answer
142
+ - utter: utter_ask_transfer_money_amount_of_money
143
+ - user: great, 50$ then
144
+ commands:
145
+ - SetSlot(transfer_money_amount_of_money, 50)
146
+ - utter: utter_ask_transfer_money_final_confirmation
147
+ - user: yes
148
+ commands:
149
+ - SetSlot(transfer_money_final_confirmation, True)
150
+ - utter: utter_transfer_complete
151
+ ```
152
+
153
+ ## Evaluation of Custom Commands
154
+
155
+ ### Defining new Custom Commands
156
+
157
+ To evaluate custom commands in Dialogue Understanding Tests, you need to define the custom command
158
+ as a subclass of `Command` and implement the `to_dsl`, `from_dsl`, and `regex_pattern` methods.
159
+
160
+ Here is an example of a custom command:
161
+
162
+ ```python
163
+ from rasa.dialogue_understanding.commands import Command
164
+
165
+
166
+ class TestCommand(Command):
167
+ ...
168
+
169
+ def to_dsl(self) -> str:
170
+ """Converts the command to a DSL string."""
171
+ return "Test()"
172
+
173
+ @classmethod
174
+ def from_dsl(cls, dsl: str) -> "TestCommand":
175
+ # Parse the DSL string and create a CustomCommand object
176
+ return TestCommand()
177
+
178
+ @staticmethod
179
+ def regex_pattern() -> str:
180
+ # Define the regex pattern that matches the DSL string
181
+ return r"Test\(\)"
182
+ ```
183
+
184
+ After defining the custom command, you can instruct the command parser to parse this new custom command
185
+ from your custom command generator by passing the custom command as an additional command in your
186
+ `parse_commands` method:
187
+
188
+ ```python
189
+ @classmethod
190
+ def parse_commands(
191
+ cls, actions: Optional[str], tracker: DialogueStateTracker, flows: FlowsList
192
+ ) -> List[Command]:
193
+ """Parse the actions returned by the llm into intent and entities.
194
+
195
+ Args:
196
+ actions: The actions returned by the llm.
197
+ tracker: The tracker containing the current state of the conversation.
198
+ flows: the list of flows
199
+
200
+ Returns:
201
+ The parsed commands.
202
+ """
203
+ from rasa.dialogue_understanding.generator.command_parser import (
204
+ parse_commands as parse_commands_using_command_parsers,
205
+ )
206
+
207
+ return parse_commands_using_command_parsers(actions, flows, additional_commands=[TestCommand])
208
+ ```
209
+
210
+ The additional commands are passed as a list of custom command classes.
211
+
212
+ When running the Dialogue Understanding Tests, you can pass the custom command as a cli argument:
213
+
214
+ ```bash
215
+ rasa test du <path-to-test-cases> --additional-commands my_module.TestCommand
216
+ ```
217
+
218
+ The `--additional-commands` argument takes a list of custom command classes separated by spaces.
219
+
220
+ ### Updating Default Commands
221
+
222
+ If you want to update the default commands, you can subclass the default command and override any of the methods.
223
+
224
+ In this example, we update the regex of the `CancelFlow` command to match the DSL string `Cancel()` instead of
225
+ `CancelFlow()`:
226
+
227
+ ```python
228
+ from rasa.dialogue_understanding.commands import CancelFlow
229
+
230
+
231
+ class CustomCancelFlow(CancelFlow):
232
+ def to_dsl(self) -> str:
233
+ """Converts the command to a DSL string."""
234
+ return "Cancel()"
235
+
236
+ @staticmethod
237
+ def regex_pattern() -> str:
238
+ # Define the regex pattern that matches the DSL string
239
+ return r"Cancel\(\)"
240
+ ```
241
+
242
+ After defining the custom command, you can instruct the command parser to parse this new custom command
243
+ from your command generator by passing the updated default command as an additional command and
244
+ removing the default command:
245
+
246
+ ```python
247
+ @classmethod
248
+ def parse_commands(
249
+ cls, actions: Optional[str], tracker: DialogueStateTracker, flows: FlowsList
250
+ ) -> List[Command]:
251
+ """Parse the actions returned by the llm into intent and entities.
252
+
253
+ Args:
254
+ actions: The actions returned by the llm.
255
+ tracker: The tracker containing the current state of the conversation.
256
+ flows: the list of flows
257
+
258
+ Returns:
259
+ The parsed commands.
260
+ """
261
+ from rasa.dialogue_understanding.generator.command_parser import (
262
+ parse_commands as parse_commands_using_command_parsers,
263
+ )
264
+
265
+ return parse_commands_using_command_parsers(actions, flows, additional_commands=[CustomCancelFlow],
266
+ remove_default_commands=[CancelFlow])
267
+ ```
268
+
269
+ When running the Dialogue Understanding Tests, you can pass the updated default command as a cli argument:
270
+
271
+ ```bash
272
+ rasa test du <path-to-test-cases> --additional-commands my_module.CustomCancelFlow --remove-default-commands CancelFlow
273
+ ```
274
+
275
+ Like the `--additional-commands` arg, the `--remove-default-commands` arg takes a list of default command classes
276
+ separated by spaces.
277
+
278
+ **Note**: The class name alone is sufficient for the `--remove-default-commands` argument because the default commands
279
+ are
280
+ already known by the Dialogue Understanding Test framework.
281
+
282
+ ## Criteria for Test Case Success
283
+
284
+ A test case is considered to have passed if all the expected commands match the predicted commands at
285
+ each step. The expected and predicted commands are considered identical if their types and arguments
286
+ exactly match, with the order of the commands being irrelevant.
287
+ To compare two commands we use the `__eq__` method of the commands.
288
+ There's an exception for the `Clarify` command.
289
+
290
+ **Clarify Command**
291
+
292
+ When defining a `Clarify` command in a Dialogue Understanding Test, you have the option to leave the
293
+ command's options empty or specify a list of options; the options are optional.
294
+ If you provide a list of options, the predicted `Clarify` command must include the exact same list
295
+ to match the expected command. If you leave the options list empty, the predicted `Clarify` command
296
+ can have any list of options.
297
+
298
+ ## Dialogue Understanding Test Output
299
+
300
+ The output of Dialogue Understanding Tests provides a comprehensive view of the chatbot's performance in
301
+ predicting and generating commands.
302
+ It includes detailed information that helps users pinpoint areas of improvement in the command generation process.
303
+ The output is logged to the console and saved in a detailed report file in a structured format for later analysis or
304
+ record-keeping.
305
+
306
+ The following information is present in the output:
307
+
308
+ - Number of passed and failed test cases.
309
+ - Number of passed and failed user utterances.
310
+ - Test case names of failed and passed test cases.
311
+ - A detailed diff of expected vs. predicted commands for each failed user message in a failed test case.
312
+ A test case can have multiple failed user messages. The command generators listed in the output are the ones that
313
+ generated the predicted commands.
314
+ Example of a failed test case diff:
315
+ ```diff
316
+ ------------- test_case: <file-path>::user_adds_contact_to_their_list -------------
317
+
318
+ Number of failed steps: 1
319
+
320
+ == failure starting at user message 'it's @barts'.
321
+
322
+ -- COMMAND GENERATOR(s) --
323
+ SingleStepLLMCommandGenerator
324
+
325
+ -- CONVERSATION --
326
+ user: I want to add someone to my contact list
327
+ bot: What's the handle of the user you want to add?
328
+ user: it's @barts
329
+ -- EXPECTED -- | -- PREDICTED --
330
+ SetSlot(handle, @barts) | SetSlot(name, @barts)
331
+ ```
332
+ - Command metrics for each command type, including the total count, true positives (tp), false positives (fp),
333
+ false negatives (fn), precision, recall, and f1-score.
334
+ Example of command metrics:
335
+ ```diff
336
+ start flow (total count: 10):
337
+ tp: 10 fp: 0 fn: 0
338
+ precision: 1.00
339
+ recall : 1.00
340
+ f1 : 1.00
341
+ ```
342
+
343
+ If you start the dialogue understanding tests with the `--output-prompt` flag, you will also see the prompt that
344
+ returned the predicted commands.
345
+
346
+ ## Converting end-to-end (E2E) Tests to Dialogue Understanding Tests
347
+
348
+ To convert end-to-end (E2E) tests into Dialogue Understanding Tests you can use a standalone Python script:
349
+
350
+ ```bash
351
+ python convert_e2e_tests_to_du_tests.py <path-to-e2e-tests>
352
+ ```
353
+
354
+ The script has the following parameters:
355
+
356
+ - `<path-to-e2e-tests>`: The path to your existing E2E test cases (can be a single file or a directory).
357
+ - `--output-folder <output-folder>`: The path where the converted test cases will be saved. The default is
358
+ `dialogue_understanding_tests`.
359
+
360
+ After running the script, the output folder structure will look like this:
361
+
362
+ ```bash
363
+ <output-folder>
364
+ |-- ready
365
+ | |-- test_case_a.yml
366
+ | |-- test_case_b.yml
367
+ | |-- ...
368
+ |-- to_review
369
+ | |-- test_case_c.yml
370
+ | |-- test_case_d.yml
371
+ | |-- ...
372
+ ```
373
+
374
+ Test cases that end up in **ready** are converted from E2E test cases that passed.
375
+ No further action is needed for these cases.
376
+ Test cases in **to_review** may require manual intervention because the E2E test failed.
377
+ Review these cases to ensure that the converted test cases are correct and the list of commands and
378
+ bot responses is complete.
379
+
@@ -0,0 +1,60 @@
1
+ from typing import List
2
+
3
+ from rasa.dialogue_understanding.commands import ClarifyCommand, Command
4
+
5
+
6
+ def are_command_lists_equal(
7
+ command_list_1: List[Command], command_list_2: List[Command]
8
+ ) -> bool:
9
+ """Check if the command lists are equal."""
10
+ if len(command_list_1) != len(command_list_2):
11
+ return False
12
+
13
+ for command in command_list_1:
14
+ if not is_command_present_in_list(command, command_list_2):
15
+ return False
16
+
17
+ return True
18
+
19
+
20
+ def is_command_present_in_list(
21
+ command: Command,
22
+ list_of_commands: List[Command],
23
+ ) -> bool:
24
+ """Check if the command is present in the list of commands."""
25
+ for command_in_list in list_of_commands:
26
+ if are_commands_equal(command, command_in_list):
27
+ return True
28
+
29
+ return False
30
+
31
+
32
+ def are_commands_equal(
33
+ command_1: Command,
34
+ command_2: Command,
35
+ ) -> bool:
36
+ """Compare the commands.
37
+
38
+ Clarify commands are compared separately as options might be optional.
39
+ """
40
+ # as options are optional for clarify commands,
41
+ # we need to check them separately
42
+ if isinstance(command_1, ClarifyCommand) and isinstance(command_2, ClarifyCommand):
43
+ return _are_clarify_commands_equal(command_1, command_2)
44
+
45
+ # an exact match is required for all other commands
46
+ return command_1 == command_2
47
+
48
+
49
+ def _are_clarify_commands_equal(
50
+ expected_command: ClarifyCommand,
51
+ predicted_command: ClarifyCommand,
52
+ ) -> bool:
53
+ # if the expected command contains options,
54
+ # the predicted command should have the same options
55
+ if expected_command.options:
56
+ return sorted(expected_command.options) == sorted(predicted_command.options)
57
+
58
+ # if the expected command does not contain options,
59
+ # it does not matter whether the predicted command has options or not
60
+ return True