rasa-pro 3.14.0.dev20250922__py3-none-any.whl → 3.14.0rc2__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 (304) hide show
  1. rasa/__main__.py +15 -3
  2. rasa/agents/__init__.py +0 -0
  3. rasa/agents/agent_factory.py +122 -0
  4. rasa/agents/agent_manager.py +211 -0
  5. rasa/agents/constants.py +43 -0
  6. rasa/agents/core/__init__.py +0 -0
  7. rasa/agents/core/agent_protocol.py +107 -0
  8. rasa/agents/core/types.py +81 -0
  9. rasa/agents/exceptions.py +38 -0
  10. rasa/agents/protocol/__init__.py +5 -0
  11. rasa/agents/protocol/a2a/__init__.py +0 -0
  12. rasa/agents/protocol/a2a/a2a_agent.py +879 -0
  13. rasa/agents/protocol/mcp/__init__.py +0 -0
  14. rasa/agents/protocol/mcp/mcp_base_agent.py +726 -0
  15. rasa/agents/protocol/mcp/mcp_open_agent.py +327 -0
  16. rasa/agents/protocol/mcp/mcp_task_agent.py +522 -0
  17. rasa/agents/schemas/__init__.py +13 -0
  18. rasa/agents/schemas/agent_input.py +38 -0
  19. rasa/agents/schemas/agent_output.py +26 -0
  20. rasa/agents/schemas/agent_tool_result.py +65 -0
  21. rasa/agents/schemas/agent_tool_schema.py +186 -0
  22. rasa/agents/templates/__init__.py +0 -0
  23. rasa/agents/templates/mcp_open_agent_prompt_template.jinja2 +20 -0
  24. rasa/agents/templates/mcp_task_agent_prompt_template.jinja2 +22 -0
  25. rasa/agents/utils.py +206 -0
  26. rasa/agents/validation.py +485 -0
  27. rasa/api.py +24 -9
  28. rasa/builder/config.py +6 -2
  29. rasa/builder/copilot/constants.py +4 -1
  30. rasa/builder/copilot/copilot.py +155 -79
  31. rasa/builder/copilot/models.py +304 -108
  32. rasa/builder/copilot/prompts/copilot_training_error_handler_prompt.jinja2 +53 -0
  33. rasa/builder/guardrails/{lakera.py → clients.py} +55 -5
  34. rasa/builder/guardrails/constants.py +3 -0
  35. rasa/builder/guardrails/models.py +45 -10
  36. rasa/builder/guardrails/policy_checker.py +324 -0
  37. rasa/builder/guardrails/utils.py +42 -276
  38. rasa/builder/jobs.py +182 -12
  39. rasa/builder/llm_service.py +32 -5
  40. rasa/builder/models.py +13 -3
  41. rasa/builder/project_generator.py +6 -1
  42. rasa/builder/service.py +31 -15
  43. rasa/builder/training_service.py +18 -24
  44. rasa/builder/validation_service.py +1 -1
  45. rasa/cli/arguments/default_arguments.py +12 -0
  46. rasa/cli/arguments/run.py +2 -0
  47. rasa/cli/arguments/train.py +2 -0
  48. rasa/cli/data.py +10 -8
  49. rasa/cli/dialogue_understanding_test.py +10 -7
  50. rasa/cli/e2e_test.py +9 -6
  51. rasa/cli/evaluate.py +4 -2
  52. rasa/cli/export.py +5 -2
  53. rasa/cli/inspect.py +8 -4
  54. rasa/cli/interactive.py +5 -4
  55. rasa/cli/llm_fine_tuning.py +11 -6
  56. rasa/cli/project_templates/finance/domain/general/help.yml +0 -0
  57. rasa/cli/project_templates/tutorial/credentials.yml +10 -0
  58. rasa/cli/run.py +12 -10
  59. rasa/cli/scaffold.py +4 -4
  60. rasa/cli/shell.py +9 -5
  61. rasa/cli/studio/studio.py +1 -1
  62. rasa/cli/test.py +34 -14
  63. rasa/cli/train.py +41 -28
  64. rasa/cli/utils.py +1 -393
  65. rasa/cli/validation/__init__.py +0 -0
  66. rasa/cli/validation/bot_config.py +223 -0
  67. rasa/cli/validation/config_path_validation.py +257 -0
  68. rasa/cli/x.py +8 -4
  69. rasa/constants.py +7 -1
  70. rasa/core/actions/action.py +51 -10
  71. rasa/core/actions/grpc_custom_action_executor.py +1 -1
  72. rasa/core/agent.py +19 -2
  73. rasa/core/available_agents.py +229 -0
  74. rasa/core/brokers/kafka.py +5 -1
  75. rasa/core/channels/__init__.py +82 -35
  76. rasa/core/channels/development_inspector.py +3 -3
  77. rasa/core/channels/inspector/README.md +25 -13
  78. rasa/core/channels/inspector/dist/assets/{arc-35222594.js → arc-6177260a.js} +1 -1
  79. rasa/core/channels/inspector/dist/assets/{blockDiagram-38ab4fdb-a0efbfd3.js → blockDiagram-38ab4fdb-b054f038.js} +1 -1
  80. rasa/core/channels/inspector/dist/assets/{c4Diagram-3d4e48cf-0584c0f2.js → c4Diagram-3d4e48cf-f25427d5.js} +1 -1
  81. rasa/core/channels/inspector/dist/assets/channel-bf9cbb34.js +1 -0
  82. rasa/core/channels/inspector/dist/assets/{classDiagram-70f12bd4-39f40dbe.js → classDiagram-70f12bd4-c7a2af53.js} +1 -1
  83. rasa/core/channels/inspector/dist/assets/{classDiagram-v2-f2320105-1ad755f3.js → classDiagram-v2-f2320105-58db65c0.js} +1 -1
  84. rasa/core/channels/inspector/dist/assets/clone-8f9083bb.js +1 -0
  85. rasa/core/channels/inspector/dist/assets/{createText-2e5e7dd3-b0f4f0fe.js → createText-2e5e7dd3-088372e2.js} +1 -1
  86. rasa/core/channels/inspector/dist/assets/{edges-e0da2a9e-9039bff9.js → edges-e0da2a9e-58676240.js} +1 -1
  87. rasa/core/channels/inspector/dist/assets/{erDiagram-9861fffd-65c9b127.js → erDiagram-9861fffd-0c14d7c6.js} +1 -1
  88. rasa/core/channels/inspector/dist/assets/{flowDb-956e92f1-4f08b38e.js → flowDb-956e92f1-ea63f85c.js} +1 -1
  89. rasa/core/channels/inspector/dist/assets/{flowDiagram-66a62f08-e95c362a.js → flowDiagram-66a62f08-a2af48cd.js} +1 -1
  90. rasa/core/channels/inspector/dist/assets/flowDiagram-v2-96b9c2cf-9ecd5b59.js +1 -0
  91. rasa/core/channels/inspector/dist/assets/{flowchart-elk-definition-4a651766-703c3015.js → flowchart-elk-definition-4a651766-6937abe7.js} +1 -1
  92. rasa/core/channels/inspector/dist/assets/{ganttDiagram-c361ad54-699328ea.js → ganttDiagram-c361ad54-7473f357.js} +1 -1
  93. rasa/core/channels/inspector/dist/assets/{gitGraphDiagram-72cf32ee-04cf4b05.js → gitGraphDiagram-72cf32ee-d0c9405e.js} +1 -1
  94. rasa/core/channels/inspector/dist/assets/{graph-ee94449e.js → graph-0a6f8466.js} +1 -1
  95. rasa/core/channels/inspector/dist/assets/{index-3862675e-940162b4.js → index-3862675e-7610671a.js} +1 -1
  96. rasa/core/channels/inspector/dist/assets/index-74e01d94.js +1354 -0
  97. rasa/core/channels/inspector/dist/assets/{infoDiagram-f8f76790-c79c2866.js → infoDiagram-f8f76790-be397dc7.js} +1 -1
  98. rasa/core/channels/inspector/dist/assets/{journeyDiagram-49397b02-84489d30.js → journeyDiagram-49397b02-4cefbf62.js} +1 -1
  99. rasa/core/channels/inspector/dist/assets/{layout-a9aa9858.js → layout-e7fbc2bf.js} +1 -1
  100. rasa/core/channels/inspector/dist/assets/{line-eb73cf26.js → line-a8aa457c.js} +1 -1
  101. rasa/core/channels/inspector/dist/assets/{linear-b3399f9a.js → linear-3351e0d2.js} +1 -1
  102. rasa/core/channels/inspector/dist/assets/{mindmap-definition-fc14e90a-b095bf1a.js → mindmap-definition-fc14e90a-b8cbf605.js} +1 -1
  103. rasa/core/channels/inspector/dist/assets/{pieDiagram-8a3498a8-07644b66.js → pieDiagram-8a3498a8-f327f774.js} +1 -1
  104. rasa/core/channels/inspector/dist/assets/{quadrantDiagram-120e2f19-573a3f9c.js → quadrantDiagram-120e2f19-2854c591.js} +1 -1
  105. rasa/core/channels/inspector/dist/assets/{requirementDiagram-deff3bca-d457e1e1.js → requirementDiagram-deff3bca-964985d5.js} +1 -1
  106. rasa/core/channels/inspector/dist/assets/{sankeyDiagram-04a897e0-9d26e1a2.js → sankeyDiagram-04a897e0-edeb4f33.js} +1 -1
  107. rasa/core/channels/inspector/dist/assets/{sequenceDiagram-704730f1-3a9cde10.js → sequenceDiagram-704730f1-fcf70125.js} +1 -1
  108. rasa/core/channels/inspector/dist/assets/{stateDiagram-587899a1-4f3e8cec.js → stateDiagram-587899a1-0e770395.js} +1 -1
  109. rasa/core/channels/inspector/dist/assets/{stateDiagram-v2-d93cdb3a-e617e5bf.js → stateDiagram-v2-d93cdb3a-af8dcd22.js} +1 -1
  110. rasa/core/channels/inspector/dist/assets/{styles-6aaf32cf-eab30d2f.js → styles-6aaf32cf-36a9e70d.js} +1 -1
  111. rasa/core/channels/inspector/dist/assets/{styles-9a916d00-09994be2.js → styles-9a916d00-884a8b5b.js} +1 -1
  112. rasa/core/channels/inspector/dist/assets/{styles-c10674c1-b7110364.js → styles-c10674c1-dc097813.js} +1 -1
  113. rasa/core/channels/inspector/dist/assets/{svgDrawCommon-08f97a94-3ebc92ad.js → svgDrawCommon-08f97a94-5a2c7eed.js} +1 -1
  114. rasa/core/channels/inspector/dist/assets/{timeline-definition-85554ec2-7d13d2f2.js → timeline-definition-85554ec2-e89c4f6e.js} +1 -1
  115. rasa/core/channels/inspector/dist/assets/{xychartDiagram-e933f94c-488385e1.js → xychartDiagram-e933f94c-afb6fe56.js} +1 -1
  116. rasa/core/channels/inspector/dist/index.html +1 -1
  117. rasa/core/channels/inspector/package.json +18 -18
  118. rasa/core/channels/inspector/src/App.tsx +29 -4
  119. rasa/core/channels/inspector/src/components/DialogueAgentStack.tsx +108 -0
  120. rasa/core/channels/inspector/src/components/{DialogueStack.tsx → DialogueHistoryStack.tsx} +4 -2
  121. rasa/core/channels/inspector/src/helpers/audio/audiostream.ts +7 -4
  122. rasa/core/channels/inspector/src/helpers/formatters.test.ts +4 -0
  123. rasa/core/channels/inspector/src/helpers/formatters.ts +24 -3
  124. rasa/core/channels/inspector/src/helpers/utils.test.ts +127 -0
  125. rasa/core/channels/inspector/src/helpers/utils.ts +66 -1
  126. rasa/core/channels/inspector/src/theme/base/styles.ts +19 -1
  127. rasa/core/channels/inspector/src/types.ts +21 -0
  128. rasa/core/channels/inspector/yarn.lock +336 -189
  129. rasa/core/channels/studio_chat.py +6 -6
  130. rasa/core/channels/telegram.py +4 -9
  131. rasa/core/channels/voice_stream/genesys.py +1 -1
  132. rasa/core/channels/voice_stream/tts/deepgram.py +140 -0
  133. rasa/core/channels/voice_stream/twilio_media_streams.py +5 -1
  134. rasa/core/channels/voice_stream/voice_channel.py +3 -0
  135. rasa/core/concurrent_lock_store.py +38 -21
  136. rasa/core/config/__init__.py +0 -0
  137. rasa/core/{available_endpoints.py → config/available_endpoints.py} +51 -16
  138. rasa/core/config/configuration.py +260 -0
  139. rasa/core/config/credentials.py +19 -0
  140. rasa/core/config/message_procesing_config.py +34 -0
  141. rasa/core/constants.py +10 -0
  142. rasa/core/iam_credentials_providers/aws_iam_credentials_providers.py +69 -4
  143. rasa/core/iam_credentials_providers/credentials_provider_protocol.py +2 -1
  144. rasa/core/lock_store.py +4 -0
  145. rasa/core/policies/enterprise_search_policy.py +5 -3
  146. rasa/core/policies/flow_policy.py +4 -4
  147. rasa/core/policies/flows/agent_executor.py +632 -0
  148. rasa/core/policies/flows/flow_executor.py +136 -75
  149. rasa/core/policies/flows/mcp_tool_executor.py +298 -0
  150. rasa/core/policies/intentless_policy.py +1 -1
  151. rasa/core/policies/ted_policy.py +20 -12
  152. rasa/core/policies/unexpected_intent_policy.py +6 -0
  153. rasa/core/processor.py +68 -44
  154. rasa/core/redis_connection_factory.py +7 -2
  155. rasa/core/run.py +37 -8
  156. rasa/core/test.py +4 -0
  157. rasa/core/tracker_stores/redis_tracker_store.py +4 -0
  158. rasa/core/tracker_stores/sql_tracker_store.py +3 -1
  159. rasa/core/tracker_stores/tracker_store.py +3 -7
  160. rasa/core/train.py +1 -1
  161. rasa/core/training/interactive.py +20 -18
  162. rasa/core/training/story_conflict.py +5 -5
  163. rasa/core/utils.py +22 -23
  164. rasa/dialogue_understanding/commands/__init__.py +8 -0
  165. rasa/dialogue_understanding/commands/cancel_flow_command.py +19 -5
  166. rasa/dialogue_understanding/commands/chit_chat_answer_command.py +21 -2
  167. rasa/dialogue_understanding/commands/clarify_command.py +20 -2
  168. rasa/dialogue_understanding/commands/continue_agent_command.py +91 -0
  169. rasa/dialogue_understanding/commands/knowledge_answer_command.py +21 -2
  170. rasa/dialogue_understanding/commands/restart_agent_command.py +162 -0
  171. rasa/dialogue_understanding/commands/start_flow_command.py +68 -7
  172. rasa/dialogue_understanding/commands/utils.py +124 -2
  173. rasa/dialogue_understanding/generator/command_parser.py +4 -0
  174. rasa/dialogue_understanding/generator/llm_based_command_generator.py +50 -12
  175. rasa/dialogue_understanding/generator/llm_command_generator.py +1 -1
  176. rasa/dialogue_understanding/generator/multi_step/multi_step_llm_command_generator.py +1 -1
  177. rasa/dialogue_understanding/generator/prompt_templates/agent_command_prompt_v2_claude_3_5_sonnet_20240620_template.jinja2 +66 -0
  178. rasa/dialogue_understanding/generator/prompt_templates/agent_command_prompt_v2_gpt_4o_2024_11_20_template.jinja2 +66 -0
  179. rasa/dialogue_understanding/generator/prompt_templates/agent_command_prompt_v3_claude_3_5_sonnet_20240620_template.jinja2 +89 -0
  180. rasa/dialogue_understanding/generator/prompt_templates/agent_command_prompt_v3_gpt_4o_2024_11_20_template.jinja2 +88 -0
  181. rasa/dialogue_understanding/generator/single_step/compact_llm_command_generator.py +42 -7
  182. rasa/dialogue_understanding/generator/single_step/search_ready_llm_command_generator.py +40 -3
  183. rasa/dialogue_understanding/generator/single_step/single_step_based_llm_command_generator.py +20 -3
  184. rasa/dialogue_understanding/patterns/cancel.py +27 -6
  185. rasa/dialogue_understanding/patterns/clarify.py +3 -14
  186. rasa/dialogue_understanding/patterns/continue_interrupted.py +239 -6
  187. rasa/dialogue_understanding/patterns/default_flows_for_patterns.yml +46 -8
  188. rasa/dialogue_understanding/processor/command_processor.py +136 -15
  189. rasa/dialogue_understanding/stack/dialogue_stack.py +98 -2
  190. rasa/dialogue_understanding/stack/frames/flow_stack_frame.py +57 -0
  191. rasa/dialogue_understanding/stack/utils.py +57 -3
  192. rasa/dialogue_understanding/utils.py +24 -4
  193. rasa/dialogue_understanding_test/du_test_runner.py +8 -3
  194. rasa/e2e_test/e2e_test_runner.py +13 -3
  195. rasa/engine/caching.py +2 -2
  196. rasa/engine/constants.py +1 -1
  197. rasa/engine/recipes/default_components.py +138 -49
  198. rasa/engine/recipes/default_recipe.py +108 -11
  199. rasa/engine/runner/dask.py +8 -5
  200. rasa/engine/validation.py +19 -6
  201. rasa/graph_components/validators/default_recipe_validator.py +86 -28
  202. rasa/hooks.py +5 -5
  203. rasa/llm_fine_tuning/utils.py +2 -2
  204. rasa/model_training.py +60 -47
  205. rasa/nlu/classifiers/diet_classifier.py +198 -98
  206. rasa/nlu/classifiers/logistic_regression_classifier.py +1 -4
  207. rasa/nlu/classifiers/mitie_intent_classifier.py +3 -0
  208. rasa/nlu/classifiers/sklearn_intent_classifier.py +1 -3
  209. rasa/nlu/extractors/crf_entity_extractor.py +9 -10
  210. rasa/nlu/extractors/mitie_entity_extractor.py +3 -0
  211. rasa/nlu/extractors/spacy_entity_extractor.py +3 -0
  212. rasa/nlu/featurizers/dense_featurizer/convert_featurizer.py +4 -0
  213. rasa/nlu/featurizers/dense_featurizer/lm_featurizer.py +5 -0
  214. rasa/nlu/featurizers/dense_featurizer/mitie_featurizer.py +2 -0
  215. rasa/nlu/featurizers/dense_featurizer/spacy_featurizer.py +3 -0
  216. rasa/nlu/featurizers/sparse_featurizer/count_vectors_featurizer.py +4 -2
  217. rasa/nlu/featurizers/sparse_featurizer/lexical_syntactic_featurizer.py +4 -0
  218. rasa/nlu/selectors/response_selector.py +10 -2
  219. rasa/nlu/tokenizers/jieba_tokenizer.py +3 -4
  220. rasa/nlu/tokenizers/mitie_tokenizer.py +3 -2
  221. rasa/nlu/tokenizers/spacy_tokenizer.py +3 -2
  222. rasa/nlu/utils/mitie_utils.py +3 -0
  223. rasa/nlu/utils/spacy_utils.py +3 -2
  224. rasa/plugin.py +8 -8
  225. rasa/privacy/privacy_manager.py +12 -3
  226. rasa/server.py +15 -3
  227. rasa/shared/agents/__init__.py +0 -0
  228. rasa/shared/agents/auth/__init__.py +0 -0
  229. rasa/shared/agents/auth/agent_auth_factory.py +105 -0
  230. rasa/shared/agents/auth/agent_auth_manager.py +92 -0
  231. rasa/shared/agents/auth/auth_strategy/__init__.py +19 -0
  232. rasa/shared/agents/auth/auth_strategy/agent_auth_strategy.py +52 -0
  233. rasa/shared/agents/auth/auth_strategy/api_key_auth_strategy.py +42 -0
  234. rasa/shared/agents/auth/auth_strategy/bearer_token_auth_strategy.py +28 -0
  235. rasa/shared/agents/auth/auth_strategy/oauth2_auth_strategy.py +167 -0
  236. rasa/shared/agents/auth/constants.py +12 -0
  237. rasa/shared/agents/auth/types.py +12 -0
  238. rasa/shared/agents/utils.py +35 -0
  239. rasa/shared/constants.py +8 -0
  240. rasa/shared/core/constants.py +16 -1
  241. rasa/shared/core/domain.py +0 -7
  242. rasa/shared/core/events.py +327 -0
  243. rasa/shared/core/flows/constants.py +5 -0
  244. rasa/shared/core/flows/flows_list.py +21 -5
  245. rasa/shared/core/flows/flows_yaml_schema.json +119 -184
  246. rasa/shared/core/flows/steps/call.py +49 -5
  247. rasa/shared/core/flows/steps/collect.py +98 -13
  248. rasa/shared/core/flows/validation.py +372 -8
  249. rasa/shared/core/flows/yaml_flows_io.py +3 -2
  250. rasa/shared/core/slots.py +2 -2
  251. rasa/shared/core/trackers.py +5 -2
  252. rasa/shared/exceptions.py +16 -0
  253. rasa/shared/importers/rasa.py +1 -1
  254. rasa/shared/importers/utils.py +9 -3
  255. rasa/shared/providers/llm/_base_litellm_client.py +41 -9
  256. rasa/shared/providers/llm/litellm_router_llm_client.py +8 -4
  257. rasa/shared/providers/llm/llm_client.py +7 -3
  258. rasa/shared/providers/llm/llm_response.py +66 -0
  259. rasa/shared/providers/llm/self_hosted_llm_client.py +8 -4
  260. rasa/shared/utils/common.py +24 -0
  261. rasa/shared/utils/health_check/health_check.py +7 -3
  262. rasa/shared/utils/llm.py +39 -16
  263. rasa/shared/utils/mcp/__init__.py +0 -0
  264. rasa/shared/utils/mcp/server_connection.py +247 -0
  265. rasa/shared/utils/mcp/utils.py +20 -0
  266. rasa/shared/utils/schemas/events.py +42 -0
  267. rasa/shared/utils/yaml.py +3 -1
  268. rasa/studio/pull/pull.py +3 -2
  269. rasa/studio/train.py +8 -7
  270. rasa/studio/upload.py +3 -6
  271. rasa/telemetry.py +69 -5
  272. rasa/tracing/config.py +45 -12
  273. rasa/tracing/constants.py +14 -0
  274. rasa/tracing/instrumentation/attribute_extractors.py +142 -9
  275. rasa/tracing/instrumentation/instrumentation.py +626 -21
  276. rasa/tracing/instrumentation/intentless_policy_instrumentation.py +4 -4
  277. rasa/tracing/instrumentation/metrics.py +32 -0
  278. rasa/tracing/metric_instrument_provider.py +68 -0
  279. rasa/utils/common.py +92 -1
  280. rasa/utils/endpoints.py +11 -2
  281. rasa/utils/log_utils.py +96 -5
  282. rasa/utils/ml_utils.py +1 -1
  283. rasa/utils/tensorflow/__init__.py +7 -0
  284. rasa/utils/tensorflow/callback.py +136 -101
  285. rasa/utils/tensorflow/crf.py +1 -1
  286. rasa/utils/tensorflow/data_generator.py +21 -8
  287. rasa/utils/tensorflow/layers.py +21 -11
  288. rasa/utils/tensorflow/metrics.py +7 -3
  289. rasa/utils/tensorflow/models.py +56 -8
  290. rasa/utils/tensorflow/rasa_layers.py +8 -6
  291. rasa/utils/tensorflow/transformer.py +2 -3
  292. rasa/utils/train_utils.py +54 -24
  293. rasa/validator.py +5 -5
  294. rasa/version.py +1 -1
  295. {rasa_pro-3.14.0.dev20250922.dist-info → rasa_pro-3.14.0rc2.dist-info}/METADATA +47 -41
  296. {rasa_pro-3.14.0.dev20250922.dist-info → rasa_pro-3.14.0rc2.dist-info}/RECORD +299 -238
  297. rasa/builder/scrape_rasa_docs.py +0 -97
  298. rasa/core/channels/inspector/dist/assets/channel-8e08bed9.js +0 -1
  299. rasa/core/channels/inspector/dist/assets/clone-78c82dea.js +0 -1
  300. rasa/core/channels/inspector/dist/assets/flowDiagram-v2-96b9c2cf-2b08f601.js +0 -1
  301. rasa/core/channels/inspector/dist/assets/index-c941dcb3.js +0 -1336
  302. {rasa_pro-3.14.0.dev20250922.dist-info → rasa_pro-3.14.0rc2.dist-info}/NOTICE +0 -0
  303. {rasa_pro-3.14.0.dev20250922.dist-info → rasa_pro-3.14.0rc2.dist-info}/WHEEL +0 -0
  304. {rasa_pro-3.14.0.dev20250922.dist-info → rasa_pro-3.14.0rc2.dist-info}/entry_points.txt +0 -0
rasa/cli/utils.py CHANGED
@@ -1,415 +1,23 @@
1
- import argparse
2
1
  import importlib
3
2
  import json
4
3
  import os
5
4
  import sys
6
- import time
7
5
  from pathlib import Path
8
6
  from types import FrameType
9
- from typing import TYPE_CHECKING, Any, Dict, List, Optional, Text, Union, overload
7
+ from typing import TYPE_CHECKING, Any, Dict, List, Text
10
8
 
11
- import randomname
12
9
  import structlog
13
10
 
14
- import rasa.shared.utils.cli
15
11
  import rasa.shared.utils.io
16
- from rasa import telemetry
17
- from rasa.exceptions import ModelNotFound, ValidationError
18
- from rasa.shared.constants import (
19
- ASSISTANT_ID_DEFAULT_VALUE,
20
- ASSISTANT_ID_KEY,
21
- DEFAULT_CONFIG_PATH,
22
- )
23
- from rasa.shared.importers.importer import TrainingDataImporter
24
- from rasa.shared.utils.common import display_research_study_prompt
25
- from rasa.shared.utils.yaml import read_config_file
26
- from rasa.utils.io import write_yaml
27
12
 
28
13
  if TYPE_CHECKING:
29
14
  from questionary import Question
30
- from typing_extensions import Literal
31
-
32
- from rasa.validator import Validator
33
15
 
34
16
  structlogger = structlog.get_logger()
35
17
 
36
18
  FREE_TEXT_INPUT_PROMPT = "Type out your own message..."
37
19
 
38
20
 
39
- @overload
40
- def get_validated_path(
41
- current: Optional[Union[Path, Text]],
42
- parameter: Text,
43
- default: Optional[Union[Path, Text, List[Text]]] = ...,
44
- none_is_valid: "Literal[False]" = ...,
45
- ) -> Union[Path, Text]: ...
46
-
47
-
48
- @overload
49
- def get_validated_path(
50
- current: Optional[Union[Path, Text]],
51
- parameter: Text,
52
- default: Optional[Union[Path, Text, List[Text]]] = ...,
53
- none_is_valid: "Literal[True]" = ...,
54
- ) -> Optional[Union[Path, Text]]: ...
55
-
56
-
57
- def get_validated_path(
58
- current: Optional[Union[Path, Text]],
59
- parameter: Text,
60
- default: Optional[Union[Path, Text, List[Text]]] = None,
61
- none_is_valid: bool = False,
62
- ) -> Optional[Union[Path, Text]]:
63
- """Checks whether a file path or its default value is valid and returns it.
64
-
65
- Args:
66
- current: The parsed value.
67
- parameter: The name of the parameter.
68
- default: one or multiple default values of the parameter.
69
- none_is_valid: `True` if `None` is valid value for the path,
70
- else `False``
71
-
72
- Returns:
73
- The current value if valid,
74
- otherwise one of the default values of the argument if valid,
75
- otherwise `None` if allowed,
76
- otherwise raises an error and exits.
77
- """
78
- if current and os.path.exists(current):
79
- return current
80
-
81
- if parameter == "model":
82
- raise ModelNotFound(
83
- f"The provided model path '{current}' could not be found. "
84
- "Provide an existing model path."
85
- )
86
-
87
- # try to find a valid option among the defaults
88
- if isinstance(default, str) or isinstance(default, Path):
89
- default_options = [str(default)]
90
- elif isinstance(default, list):
91
- default_options = default
92
- else:
93
- default_options = []
94
-
95
- valid_options = (option for option in default_options if os.path.exists(option))
96
- chosen_option = next(valid_options, None)
97
-
98
- # warn and log if user-chosen parameter wasn't found and thus overwritten
99
- if chosen_option:
100
- shared_info = f"Using default location '{chosen_option}' instead."
101
- if current is None:
102
- structlogger.debug(
103
- "cli.get_validated_path.parameter_not_set",
104
- parameter=parameter,
105
- event_info=(f"Parameter '{parameter}' was not set. {shared_info}"),
106
- )
107
- elif current not in default_options:
108
- structlogger.warn(
109
- "cli.get_validated_path.path_does_not_exists",
110
- path=current,
111
- event_info=(
112
- f"The path '{current}' does not seem to exist. {shared_info}"
113
- ),
114
- )
115
-
116
- if chosen_option is None and not none_is_valid:
117
- cancel_cause_not_found(current, parameter, default)
118
-
119
- return chosen_option
120
-
121
-
122
- def missing_config_keys(
123
- path: Union["Path", Text], mandatory_keys: List[Text]
124
- ) -> List[Text]:
125
- """Checks whether the config file at `path` contains the `mandatory_keys`.
126
-
127
- Args:
128
- path: The path to the config file.
129
- mandatory_keys: A list of mandatory config keys.
130
-
131
- Returns:
132
- The list of missing config keys.
133
- """
134
- if not os.path.exists(path):
135
- return mandatory_keys
136
-
137
- config_data = read_config_file(path)
138
-
139
- return [k for k in mandatory_keys if k not in config_data or config_data[k] is None]
140
-
141
-
142
- def validate_assistant_id_in_config(config_file: Union["Path", Text]) -> None:
143
- """Verifies that the assistant_id key exists and has a unique value in config.
144
-
145
- Issues a warning if the key does not exist or has the default value and replaces it
146
- with a pseudo-random string value.
147
- """
148
- config_data = read_config_file(config_file, reader_type=["safe", "rt"])
149
- assistant_id = config_data.get(ASSISTANT_ID_KEY)
150
-
151
- if assistant_id is None or assistant_id == ASSISTANT_ID_DEFAULT_VALUE:
152
- structlogger.warn(
153
- "cli.validate_assistant_id_in_config.missing_unique_assistant_id_key",
154
- config=config_file,
155
- missing_key=ASSISTANT_ID_KEY,
156
- event_info=(
157
- f"The config file '{config_file!s}' is "
158
- f"missing a unique value for the "
159
- f"'{ASSISTANT_ID_KEY}' mandatory key. "
160
- f"Proceeding with generating a random "
161
- f"value and overwriting the '{ASSISTANT_ID_KEY}'"
162
- f" in the config file."
163
- ),
164
- )
165
-
166
- # add random value for assistant id, overwrite config file
167
- time_format = "%Y%m%d-%H%M%S"
168
- config_data[ASSISTANT_ID_KEY] = (
169
- f"{time.strftime(time_format)}-{randomname.get_name()}"
170
- )
171
-
172
- write_yaml(data=config_data, target=config_file, should_preserve_key_order=True)
173
-
174
- return
175
-
176
-
177
- def validate_config_path(
178
- config: Optional[Union[Text, "Path"]],
179
- default_config: Text = DEFAULT_CONFIG_PATH,
180
- ) -> Text:
181
- """Verifies that the config path exists.
182
-
183
- Exit if the config file does not exist.
184
-
185
- Args:
186
- config: Path to the config file.
187
- default_config: default config to use if the file at `config` doesn't exist.
188
-
189
- Returns: The path to the config file.
190
- """
191
- config = rasa.cli.utils.get_validated_path(config, "config", default_config)
192
-
193
- if not config or not os.path.exists(config):
194
- display_research_study_prompt()
195
- raise ValidationError(
196
- code="cli.validate_config_path.does_not_exists",
197
- config=config,
198
- event_info=(
199
- f"The config file '{config}' does not exist. "
200
- f"Use '--config' to specify a valid config file."
201
- ),
202
- )
203
-
204
- return str(config)
205
-
206
-
207
- def validate_mandatory_config_keys(
208
- config: Union[Text, "Path"],
209
- mandatory_keys: List[Text],
210
- ) -> Text:
211
- """Get a config from a config file and check if it is valid.
212
-
213
- Exit if the config isn't valid.
214
-
215
- Args:
216
- config: Path to the config file.
217
- mandatory_keys: The keys that have to be specified in the config file.
218
-
219
- Returns: The path to the config file if the config is valid.
220
- """
221
- missing_keys = set(rasa.cli.utils.missing_config_keys(config, mandatory_keys))
222
- if missing_keys:
223
- display_research_study_prompt()
224
- raise ValidationError(
225
- code="cli.validate_mandatory_config_keys.missing_keys",
226
- config=config,
227
- missing_keys=missing_keys,
228
- event_info=(
229
- "The config file '{}' is missing mandatory parameters: "
230
- "'{}'. Add missing parameters to config file and try again.".format(
231
- config, "', '".join(missing_keys)
232
- )
233
- ),
234
- )
235
-
236
- return str(config)
237
-
238
-
239
- def get_validated_config(
240
- config: Optional[Union[Text, "Path"]],
241
- mandatory_keys: List[Text],
242
- default_config: Text = DEFAULT_CONFIG_PATH,
243
- ) -> Text:
244
- """Validates config and returns path to validated config file."""
245
- config = validate_config_path(config, default_config)
246
- validate_assistant_id_in_config(config)
247
-
248
- config = validate_mandatory_config_keys(config, mandatory_keys)
249
-
250
- return config
251
-
252
-
253
- def validate_files(
254
- fail_on_warnings: bool,
255
- max_history: Optional[int],
256
- importer: TrainingDataImporter,
257
- stories_only: bool = False,
258
- flows_only: bool = False,
259
- translations_only: bool = False,
260
- ) -> None:
261
- """Validates either the story structure or the entire project.
262
-
263
- Args:
264
- fail_on_warnings: `True` if the process should exit with a non-zero status
265
- max_history: The max history to use when validating the story structure.
266
- importer: The `TrainingDataImporter` to use to load the training data.
267
- stories_only: If `True`, only the story structure is validated.
268
- flows_only: If `True`, only the flows are validated.
269
- translations_only: If `True`, only the translations data is validated.
270
- """
271
- from rasa.validator import Validator
272
-
273
- validator = Validator.from_importer(importer)
274
-
275
- if stories_only:
276
- all_good = _validate_story_structure(validator, max_history, fail_on_warnings)
277
- elif flows_only:
278
- all_good = validator.verify_flows()
279
- elif translations_only:
280
- all_good = validator.verify_translations()
281
- else:
282
- if importer.get_domain().is_empty():
283
- structlogger.error(
284
- "cli.validate_files.empty_domain",
285
- event_info="Encountered empty domain during validation.",
286
- )
287
- display_research_study_prompt()
288
- raise ValidationError(
289
- code="cli.validate_files.empty_domain",
290
- event_info="Encountered empty domain during validation.",
291
- )
292
-
293
- valid_domain = _validate_domain(validator)
294
- valid_nlu = _validate_nlu(validator, fail_on_warnings)
295
- valid_stories = _validate_story_structure(
296
- validator, max_history, fail_on_warnings
297
- )
298
- valid_flows = validator.verify_flows()
299
- if validator.config:
300
- valid_translations = validator.verify_translations(summary_mode=True)
301
- else:
302
- valid_translations = True
303
- valid_CALM_slot_mappings = validator.validate_CALM_slot_mappings()
304
-
305
- all_good = (
306
- valid_domain
307
- and valid_nlu
308
- and valid_stories
309
- and valid_flows
310
- and valid_translations
311
- and valid_CALM_slot_mappings
312
- )
313
-
314
- if validator.config:
315
- validator.warn_if_config_mandatory_keys_are_not_set()
316
-
317
- telemetry.track_validate_files(all_good)
318
- if not all_good:
319
- structlogger.error(
320
- "cli.validate_files.project_validation_error",
321
- event_info="Project validation completed with errors.",
322
- )
323
- display_research_study_prompt()
324
- raise ValidationError(
325
- code="cli.validate_files.project_validation_error",
326
- event_info="Project validation completed with errors.",
327
- )
328
-
329
-
330
- def _validate_domain(validator: "Validator") -> bool:
331
- valid_domain_validity = validator.verify_domain_validity()
332
- valid_actions_in_stories_rules = validator.verify_actions_in_stories_rules()
333
- valid_forms_in_stories_rules = validator.verify_forms_in_stories_rules()
334
- valid_form_slots = validator.verify_form_slots()
335
- valid_slot_mappings = validator.verify_slot_mappings()
336
- valid_responses = validator.check_for_no_empty_parenthesis_in_responses()
337
- valid_buttons = validator.validate_button_payloads()
338
- valid_slot_validation = validator.verify_slot_validation()
339
- valid_conditional_responses = (
340
- validator.validate_conditional_response_variation_predicates()
341
- )
342
- return (
343
- valid_domain_validity
344
- and valid_actions_in_stories_rules
345
- and valid_forms_in_stories_rules
346
- and valid_form_slots
347
- and valid_slot_mappings
348
- and valid_responses
349
- and valid_buttons
350
- and valid_slot_validation
351
- and valid_conditional_responses
352
- )
353
-
354
-
355
- def _validate_nlu(validator: "Validator", fail_on_warnings: bool) -> bool:
356
- return validator.verify_nlu(not fail_on_warnings)
357
-
358
-
359
- def _validate_story_structure(
360
- validator: "Validator", max_history: Optional[int], fail_on_warnings: bool
361
- ) -> bool:
362
- # Check if a valid setting for `max_history` was given
363
- if isinstance(max_history, int) and max_history < 1:
364
- raise argparse.ArgumentTypeError(
365
- f"The value of `--max-history {max_history}` is not a positive integer."
366
- )
367
-
368
- return validator.verify_story_structure(
369
- not fail_on_warnings, max_history=max_history
370
- )
371
-
372
-
373
- def cancel_cause_not_found(
374
- current: Optional[Union["Path", Text]],
375
- parameter: Text,
376
- default: Optional[Union["Path", Text, List[Text]]],
377
- ) -> None:
378
- """Exits with an error because the given path was not valid.
379
-
380
- Args:
381
- current: The path given by the user.
382
- parameter: The name of the parameter.
383
- default: The default value of the parameter.
384
-
385
- """
386
- default_clause = ""
387
- if default and isinstance(default, str):
388
- default_clause = f"use the default location ('{default}') or"
389
- elif default and isinstance(default, list):
390
- default_clause = f"use one of the default locations ({', '.join(default)}) or"
391
-
392
- structlogger.error(
393
- "cli.path_does_not_exist",
394
- path=current,
395
- event_info=(
396
- f"The path '{current}' does not exist. "
397
- f"Please make sure to {default_clause} specify it "
398
- f"with '--{parameter}'."
399
- ),
400
- )
401
- display_research_study_prompt()
402
- raise ValidationError(
403
- code="cli.path_does_not_exist",
404
- path=current,
405
- event_info=(
406
- f"The path '{current}' does not exist. "
407
- f"Please make sure to {default_clause} specify it "
408
- f"with '--{parameter}'."
409
- ),
410
- )
411
-
412
-
413
21
  def parse_last_positional_argument_as_model_path() -> None:
414
22
  """Fixes the parsing of a potential positional model path argument."""
415
23
  if (
File without changes
@@ -0,0 +1,223 @@
1
+ import argparse
2
+ import os
3
+ from typing import TYPE_CHECKING, Optional
4
+
5
+ import structlog
6
+
7
+ from rasa import telemetry
8
+ from rasa.exceptions import ValidationError
9
+ from rasa.shared.importers.importer import TrainingDataImporter
10
+ from rasa.shared.utils.common import display_research_study_prompt
11
+
12
+ if TYPE_CHECKING:
13
+ from rasa.validator import Validator
14
+
15
+ structlogger = structlog.get_logger()
16
+
17
+ FREE_TEXT_INPUT_PROMPT = "Type out your own message..."
18
+
19
+
20
+ def _validate_domain(validator: "Validator") -> bool:
21
+ valid_domain_validity = validator.verify_domain_validity()
22
+ valid_actions_in_stories_rules = validator.verify_actions_in_stories_rules()
23
+ valid_forms_in_stories_rules = validator.verify_forms_in_stories_rules()
24
+ valid_form_slots = validator.verify_form_slots()
25
+ valid_slot_mappings = validator.verify_slot_mappings()
26
+ valid_responses = validator.check_for_no_empty_parenthesis_in_responses()
27
+ valid_buttons = validator.validate_button_payloads()
28
+ valid_slot_validation = validator.verify_slot_validation()
29
+ valid_conditional_responses = (
30
+ validator.validate_conditional_response_variation_predicates()
31
+ )
32
+ return (
33
+ valid_domain_validity
34
+ and valid_actions_in_stories_rules
35
+ and valid_forms_in_stories_rules
36
+ and valid_form_slots
37
+ and valid_slot_mappings
38
+ and valid_responses
39
+ and valid_buttons
40
+ and valid_slot_validation
41
+ and valid_conditional_responses
42
+ )
43
+
44
+
45
+ def _validate_nlu(validator: "Validator", fail_on_warnings: bool) -> bool:
46
+ return validator.verify_nlu(not fail_on_warnings)
47
+
48
+
49
+ def _validate_story_structure(
50
+ validator: "Validator", max_history: Optional[int], fail_on_warnings: bool
51
+ ) -> bool:
52
+ # Check if a valid setting for `max_history` was given
53
+ if isinstance(max_history, int) and max_history < 1:
54
+ raise argparse.ArgumentTypeError(
55
+ f"The value of `--max-history {max_history}` is not a positive integer."
56
+ )
57
+
58
+ return validator.verify_story_structure(
59
+ not fail_on_warnings, max_history=max_history
60
+ )
61
+
62
+
63
+ def _validate_sub_agents(sub_agents_path: str) -> bool:
64
+ """Validates sub-agents configuration.
65
+
66
+ Args:
67
+ sub_agents_path: Path to the sub-agents directory.
68
+
69
+ Returns:
70
+ True if validation passes, False otherwise.
71
+ """
72
+ from rasa.agents.validation import validate_agent_folder
73
+ from rasa.core.constants import DEFAULT_SUB_AGENTS
74
+
75
+ try:
76
+ # Check if the sub-agents directory exists
77
+ if not os.path.isdir(sub_agents_path):
78
+ # If the sub-agents-path points to the default folder and it doesn't exist,
79
+ # no agents are available.
80
+ # AvailableAgents will handle the non-existing folder gracefully.
81
+
82
+ if sub_agents_path == DEFAULT_SUB_AGENTS:
83
+ structlogger.info(
84
+ "cli.validate_files.sub_agents_validation",
85
+ sub_agents_path=sub_agents_path,
86
+ event_info="Default sub-agents directory does not exist, "
87
+ "no sub-agents will be available.",
88
+ )
89
+ return True
90
+ else:
91
+ # For user-specified paths, the directory must exist
92
+ structlogger.error(
93
+ "cli.validate_files.sub_agents_validation_error",
94
+ sub_agents_path=sub_agents_path,
95
+ event_info=f"Sub-agents directory '{sub_agents_path}' "
96
+ "does not exist.",
97
+ )
98
+ return False
99
+
100
+ # Validate the actual config content using AvailableAgents
101
+ # This will validate file existence, structure, mandatory keys, etc.
102
+ try:
103
+ validate_agent_folder(sub_agents_path)
104
+
105
+ except ValidationError as e:
106
+ # This is a validation error - log it and return False
107
+ structlogger.error(
108
+ "cli.validate_files.sub_agents_validation_error",
109
+ sub_agents_path=sub_agents_path,
110
+ validation_error=str(e),
111
+ event_info=f"Sub-agents configuration validation failed: {e}",
112
+ )
113
+ return False
114
+
115
+ except Exception as e:
116
+ # This is an unexpected error
117
+ structlogger.error(
118
+ "cli.validate_files.sub_agents_validation_error",
119
+ sub_agents_path=sub_agents_path,
120
+ error=str(e),
121
+ event_info=f"Unexpected error during sub-agents validation: {e}",
122
+ )
123
+ return False
124
+
125
+ structlogger.info(
126
+ "cli.validate_files.sub_agents_validation_success",
127
+ sub_agents_path=sub_agents_path,
128
+ event_info="Sub-agents validation passed successfully.",
129
+ )
130
+ return True
131
+
132
+ except Exception as e:
133
+ # This is an unexpected error
134
+ structlogger.error(
135
+ "cli.validate_files.sub_agents_validation_error",
136
+ sub_agents_path=sub_agents_path,
137
+ error=str(e),
138
+ event_info="Sub-agents validation failed.",
139
+ )
140
+ return False
141
+
142
+
143
+ def validate_files(
144
+ fail_on_warnings: bool,
145
+ max_history: Optional[int],
146
+ importer: TrainingDataImporter,
147
+ stories_only: bool = False,
148
+ flows_only: bool = False,
149
+ translations_only: bool = False,
150
+ sub_agents: Optional[str] = None,
151
+ ) -> None:
152
+ """Validates either the story structure or the entire project.
153
+
154
+ Args:
155
+ fail_on_warnings: `True` if the process should exit with a non-zero status
156
+ max_history: The max history to use when validating the story structure.
157
+ importer: The `TrainingDataImporter` to use to load the training data.
158
+ stories_only: If `True`, only the story structure is validated.
159
+ flows_only: If `True`, only the flows are validated.
160
+ translations_only: If `True`, only the translations data is validated.
161
+ sub_agents: Path to sub-agents directory for validation.
162
+ """
163
+ from rasa.validator import Validator
164
+
165
+ validator = Validator.from_importer(importer)
166
+
167
+ if stories_only:
168
+ all_good = _validate_story_structure(validator, max_history, fail_on_warnings)
169
+ elif flows_only:
170
+ all_good = validator.verify_flows()
171
+ elif translations_only:
172
+ all_good = validator.verify_translations()
173
+ else:
174
+ if importer.get_domain().is_empty():
175
+ structlogger.error(
176
+ "cli.validate_files.empty_domain",
177
+ event_info="Encountered empty domain during validation.",
178
+ )
179
+ display_research_study_prompt()
180
+ raise ValidationError(
181
+ code="cli.validate_files.empty_domain",
182
+ event_info="Encountered empty domain during validation.",
183
+ )
184
+
185
+ valid_domain = _validate_domain(validator)
186
+ valid_nlu = _validate_nlu(validator, fail_on_warnings)
187
+ valid_stories = _validate_story_structure(
188
+ validator, max_history, fail_on_warnings
189
+ )
190
+ valid_flows = validator.verify_flows()
191
+ if validator.config:
192
+ valid_translations = validator.verify_translations(summary_mode=True)
193
+ else:
194
+ valid_translations = True
195
+ valid_CALM_slot_mappings = validator.validate_CALM_slot_mappings()
196
+
197
+ # Validate sub-agents if specified
198
+ valid_sub_agents = _validate_sub_agents(sub_agents) if sub_agents else True
199
+
200
+ all_good = (
201
+ valid_domain
202
+ and valid_nlu
203
+ and valid_stories
204
+ and valid_flows
205
+ and valid_translations
206
+ and valid_CALM_slot_mappings
207
+ and valid_sub_agents
208
+ )
209
+
210
+ if validator.config:
211
+ validator.warn_if_config_mandatory_keys_are_not_set()
212
+
213
+ telemetry.track_validate_files(all_good)
214
+ if not all_good:
215
+ structlogger.error(
216
+ "cli.validate_files.project_validation_error",
217
+ event_info="Project validation completed with errors.",
218
+ )
219
+ display_research_study_prompt()
220
+ raise ValidationError(
221
+ code="cli.validate_files.project_validation_error",
222
+ event_info="Project validation completed with errors.",
223
+ )