rasa-pro 3.14.0.dev20250818__py3-none-any.whl → 3.14.0.dev20250901__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 (361) hide show
  1. rasa/builder/README.md +120 -0
  2. rasa/builder/__init__.py +0 -0
  3. rasa/builder/auth.py +176 -0
  4. rasa/builder/config.py +91 -0
  5. rasa/builder/copilot/__init__.py +0 -0
  6. rasa/builder/copilot/constants.py +28 -0
  7. rasa/builder/copilot/copilot.py +376 -0
  8. rasa/builder/copilot/copilot_response_handler.py +522 -0
  9. rasa/builder/copilot/copilot_templated_message_provider.py +58 -0
  10. rasa/builder/copilot/exceptions.py +32 -0
  11. rasa/builder/copilot/models.py +464 -0
  12. rasa/builder/copilot/prompts/__init__.py +0 -0
  13. rasa/builder/copilot/prompts/copilot_system_prompt.jinja2 +771 -0
  14. rasa/builder/copilot/signing.py +305 -0
  15. rasa/builder/copilot/telemetry.py +200 -0
  16. rasa/builder/copilot/templated_messages/__init__.py +0 -0
  17. rasa/builder/copilot/templated_messages/copilot_internal_messages_templates.yml +16 -0
  18. rasa/builder/copilot/templated_messages/copilot_templated_responses.yml +38 -0
  19. rasa/builder/document_retrieval/__init__.py +0 -0
  20. rasa/builder/document_retrieval/constants.py +15 -0
  21. rasa/builder/document_retrieval/inkeep-rag-response-schema.json +64 -0
  22. rasa/builder/document_retrieval/inkeep_document_retrieval.py +238 -0
  23. rasa/builder/document_retrieval/models.py +62 -0
  24. rasa/builder/download.py +147 -0
  25. rasa/builder/exceptions.py +91 -0
  26. rasa/builder/guardrails/__init__.py +1 -0
  27. rasa/builder/guardrails/constants.py +9 -0
  28. rasa/builder/guardrails/exceptions.py +4 -0
  29. rasa/builder/guardrails/lakera.py +206 -0
  30. rasa/builder/guardrails/models.py +231 -0
  31. rasa/builder/guardrails/store.py +238 -0
  32. rasa/builder/guardrails/utils.py +328 -0
  33. rasa/builder/job_manager.py +87 -0
  34. rasa/builder/jobs.py +270 -0
  35. rasa/builder/llm_service.py +246 -0
  36. rasa/builder/logging_utils.py +265 -0
  37. rasa/builder/main.py +258 -0
  38. rasa/builder/models.py +216 -0
  39. rasa/builder/project_generator.py +462 -0
  40. rasa/builder/project_info.py +72 -0
  41. rasa/builder/scrape_rasa_docs.py +97 -0
  42. rasa/builder/service.py +1335 -0
  43. rasa/builder/shared/tracker_context.py +212 -0
  44. rasa/builder/skill_to_bot_prompt.jinja2 +164 -0
  45. rasa/builder/training_service.py +124 -0
  46. rasa/builder/validation_service.py +97 -0
  47. rasa/cli/project_templates/basic/actions/__init__ +0 -0
  48. rasa/cli/project_templates/basic/actions/action_human_handoff.py +40 -0
  49. rasa/cli/project_templates/basic/config.yml +27 -0
  50. rasa/cli/project_templates/basic/credentials.yml +33 -0
  51. rasa/cli/project_templates/basic/data/data.md +9 -0
  52. rasa/cli/project_templates/basic/data/general/feedback.yml +21 -0
  53. rasa/cli/project_templates/basic/data/general/goodbye.yml +6 -0
  54. rasa/cli/project_templates/basic/data/general/hello.yml +6 -0
  55. rasa/cli/project_templates/basic/data/general/help.yml +6 -0
  56. rasa/cli/project_templates/basic/data/general/human_handoff.yml +16 -0
  57. rasa/cli/project_templates/basic/data/general/show_faqs.yml +6 -0
  58. rasa/cli/project_templates/basic/data/system/patterns/pattern_cannot_handle.yml +7 -0
  59. rasa/cli/project_templates/basic/data/system/patterns/pattern_completed.yml +7 -0
  60. rasa/cli/project_templates/basic/data/system/patterns/pattern_correction.yml +7 -0
  61. rasa/cli/project_templates/basic/data/system/patterns/pattern_search.yml +8 -0
  62. rasa/cli/project_templates/basic/data/system/patterns/pattern_session_start.yml +8 -0
  63. rasa/cli/project_templates/basic/docs/docs.md +5 -0
  64. rasa/cli/project_templates/basic/docs/template.txt +28 -0
  65. rasa/cli/project_templates/basic/domain/domain.md +9 -0
  66. rasa/cli/project_templates/basic/domain/general/feedback.yml +25 -0
  67. rasa/cli/project_templates/basic/domain/general/goodbye.yml +9 -0
  68. rasa/cli/project_templates/basic/domain/general/hello.yml +7 -0
  69. rasa/cli/project_templates/basic/domain/general/help.yml +21 -0
  70. rasa/cli/project_templates/basic/domain/general/human_handoff.yml +32 -0
  71. rasa/cli/project_templates/basic/domain/general/show_faqs.yml +14 -0
  72. rasa/cli/project_templates/basic/domain/system/patterns/pattern_cannot_handle.yml +5 -0
  73. rasa/cli/project_templates/basic/domain/system/patterns/pattern_session_start.yml +19 -0
  74. rasa/cli/project_templates/basic/endpoints.yml +63 -0
  75. rasa/cli/project_templates/basic/prompts/rephraser_demo_personality_prompt.jinja2 +38 -0
  76. rasa/cli/project_templates/finance/actions/__init__.py +46 -0
  77. rasa/cli/project_templates/finance/actions/accounts/__init__.py +0 -0
  78. rasa/cli/project_templates/finance/actions/accounts/action_ask_account.py +47 -0
  79. rasa/cli/project_templates/finance/actions/accounts/action_check_balance.py +40 -0
  80. rasa/cli/project_templates/finance/actions/action_session_start.py +74 -0
  81. rasa/cli/project_templates/finance/actions/cards/__init__.py +0 -0
  82. rasa/cli/project_templates/finance/actions/cards/action_ask_card.py +48 -0
  83. rasa/cli/project_templates/finance/actions/cards/action_check_card_existence.py +36 -0
  84. rasa/cli/project_templates/finance/actions/cards/action_update_card_status.py +54 -0
  85. rasa/cli/project_templates/finance/actions/database.py +277 -0
  86. rasa/cli/project_templates/finance/actions/transfers/__init__.py +0 -0
  87. rasa/cli/project_templates/finance/actions/transfers/action_add_payee.py +52 -0
  88. rasa/cli/project_templates/finance/actions/transfers/action_ask_account_from.py +51 -0
  89. rasa/cli/project_templates/finance/actions/transfers/action_check_payee_existence.py +40 -0
  90. rasa/cli/project_templates/finance/actions/transfers/action_check_sufficient_funds.py +40 -0
  91. rasa/cli/project_templates/finance/actions/transfers/action_list_payees.py +46 -0
  92. rasa/cli/project_templates/finance/actions/transfers/action_process_immediate_payment.py +18 -0
  93. rasa/cli/project_templates/finance/actions/transfers/action_remove_payee.py +49 -0
  94. rasa/cli/project_templates/finance/actions/transfers/action_schedule_payment.py +19 -0
  95. rasa/cli/project_templates/finance/actions/transfers/action_validate_payment_date.py +36 -0
  96. rasa/cli/project_templates/finance/config.yml +21 -0
  97. rasa/cli/project_templates/finance/credentials.yml +32 -0
  98. rasa/cli/project_templates/finance/csvs/accounts.csv +8 -0
  99. rasa/cli/project_templates/finance/csvs/advisors.csv +7 -0
  100. rasa/cli/project_templates/finance/csvs/appointments.csv +211 -0
  101. rasa/cli/project_templates/finance/csvs/branches.csv +10 -0
  102. rasa/cli/project_templates/finance/csvs/cards.csv +11 -0
  103. rasa/cli/project_templates/finance/csvs/payees.csv +11 -0
  104. rasa/cli/project_templates/finance/csvs/transactions.csv +71 -0
  105. rasa/cli/project_templates/finance/csvs/users.csv +4 -0
  106. rasa/cli/project_templates/finance/data/accounts/check_balance.yml +10 -0
  107. rasa/cli/project_templates/finance/data/cards/block_card.yml +66 -0
  108. rasa/cli/project_templates/finance/data/cards/select_card.yml +12 -0
  109. rasa/cli/project_templates/finance/data/general/bot_identity.yml +6 -0
  110. rasa/cli/project_templates/finance/data/general/feedback.yml +20 -0
  111. rasa/cli/project_templates/finance/data/general/goodbye.yml +6 -0
  112. rasa/cli/project_templates/finance/data/general/hello.yml +7 -0
  113. rasa/cli/project_templates/finance/data/general/help.yml +9 -0
  114. rasa/cli/project_templates/finance/data/general/human_handoff.yml +16 -0
  115. rasa/cli/project_templates/finance/data/general/welcome.yml +9 -0
  116. rasa/cli/project_templates/finance/data/system/patterns/pattern_chitchat.yml +5 -0
  117. rasa/cli/project_templates/finance/data/system/patterns/pattern_completed.yml +7 -0
  118. rasa/cli/project_templates/finance/data/system/patterns/pattern_correction.yml +7 -0
  119. rasa/cli/project_templates/finance/data/system/patterns/pattern_search.yml +8 -0
  120. rasa/cli/project_templates/finance/data/system/patterns/pattern_session_start.yml +8 -0
  121. rasa/cli/project_templates/finance/data/system/source/accounts.json +51 -0
  122. rasa/cli/project_templates/finance/data/system/source/advisors.json +44 -0
  123. rasa/cli/project_templates/finance/data/system/source/appointments.json +1474 -0
  124. rasa/cli/project_templates/finance/data/system/source/branches.json +47 -0
  125. rasa/cli/project_templates/finance/data/system/source/cards.json +72 -0
  126. rasa/cli/project_templates/finance/data/system/source/payees.json +74 -0
  127. rasa/cli/project_templates/finance/data/system/source/transactions.json +492 -0
  128. rasa/cli/project_templates/finance/data/system/source/users.json +29 -0
  129. rasa/cli/project_templates/finance/data/transfers/add_payee.yml +29 -0
  130. rasa/cli/project_templates/finance/data/transfers/list_payees.yml +5 -0
  131. rasa/cli/project_templates/finance/data/transfers/remove_payee.yml +21 -0
  132. rasa/cli/project_templates/finance/data/transfers/transfer_money.yml +67 -0
  133. rasa/cli/project_templates/finance/docs/bank_of_rasa_faq/block_card/consequences_of_blocking_card.txt +8 -0
  134. rasa/cli/project_templates/finance/docs/bank_of_rasa_faq/block_card/reasons_to_block_card.txt +8 -0
  135. rasa/cli/project_templates/finance/docs/bank_of_rasa_faq/block_card/recovering_from_card_fraud.txt +8 -0
  136. rasa/cli/project_templates/finance/docs/bank_of_rasa_faq/block_card/tips_for_card_security.txt +8 -0
  137. rasa/cli/project_templates/finance/docs/bank_of_rasa_faq/block_card/what_to_do_if_card_is_lost.txt +8 -0
  138. rasa/cli/project_templates/finance/docs/bank_of_rasa_faq/check_balance/account_balance_security.txt +7 -0
  139. rasa/cli/project_templates/finance/docs/bank_of_rasa_faq/check_balance/common_balance_inquiries.txt +8 -0
  140. rasa/cli/project_templates/finance/docs/bank_of_rasa_faq/check_balance/methods_to_check_balance.txt +8 -0
  141. rasa/cli/project_templates/finance/docs/bank_of_rasa_faq/check_balance/understanding_balance_updates.txt +8 -0
  142. rasa/cli/project_templates/finance/docs/bank_of_rasa_faq/check_balance/what_to_do_if_balance_is_incorrect.txt +8 -0
  143. rasa/cli/project_templates/finance/docs/bank_of_rasa_faq/manage_payees/benefits_of_authorised_payees.txt +8 -0
  144. rasa/cli/project_templates/finance/docs/bank_of_rasa_faq/manage_payees/common_issues_with_payees.txt +8 -0
  145. rasa/cli/project_templates/finance/docs/bank_of_rasa_faq/manage_payees/general_payee_information.txt +8 -0
  146. rasa/cli/project_templates/finance/docs/bank_of_rasa_faq/manage_payees/payee_management_tips.txt +8 -0
  147. rasa/cli/project_templates/finance/docs/bank_of_rasa_faq/manage_payees/understanding_payee_types.txt +8 -0
  148. rasa/cli/project_templates/finance/docs/bank_of_rasa_faq/transfer_money/common_transfer_errors.txt +8 -0
  149. rasa/cli/project_templates/finance/docs/bank_of_rasa_faq/transfer_money/fees_for_transfers.txt +8 -0
  150. rasa/cli/project_templates/finance/docs/bank_of_rasa_faq/transfer_money/general_transfer_information.txt +8 -0
  151. rasa/cli/project_templates/finance/docs/bank_of_rasa_faq/transfer_money/security_tips_for_transfers.txt +8 -0
  152. rasa/cli/project_templates/finance/docs/bank_of_rasa_faq/transfer_money/transfer_processing_times.txt +8 -0
  153. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part1.txt +50 -0
  154. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part10.txt +50 -0
  155. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part11.txt +48 -0
  156. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part12.txt +50 -0
  157. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part13.txt +50 -0
  158. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part14.txt +47 -0
  159. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part15.txt +50 -0
  160. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part16.txt +50 -0
  161. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part17.txt +47 -0
  162. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part18.txt +50 -0
  163. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part19.txt +50 -0
  164. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part2.txt +50 -0
  165. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part20.txt +47 -0
  166. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part21.txt +50 -0
  167. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part22.txt +50 -0
  168. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part23.txt +47 -0
  169. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part24.txt +50 -0
  170. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part25.txt +50 -0
  171. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part26.txt +47 -0
  172. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part27.txt +50 -0
  173. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part28.txt +50 -0
  174. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part29.txt +47 -0
  175. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part3.txt +47 -0
  176. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part30.txt +50 -0
  177. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part31.txt +50 -0
  178. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part32.txt +47 -0
  179. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part33.txt +50 -0
  180. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part34.txt +50 -0
  181. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part35.txt +47 -0
  182. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part36.txt +50 -0
  183. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part37.txt +50 -0
  184. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part38.txt +47 -0
  185. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part39.txt +50 -0
  186. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part4.txt +50 -0
  187. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part40.txt +50 -0
  188. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part41.txt +47 -0
  189. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part42.txt +50 -0
  190. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part43.txt +50 -0
  191. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part44.txt +47 -0
  192. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part45.txt +50 -0
  193. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part46.txt +50 -0
  194. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part47.txt +47 -0
  195. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part48.txt +50 -0
  196. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part49.txt +50 -0
  197. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part5.txt +50 -0
  198. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part50.txt +47 -0
  199. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part51.txt +50 -0
  200. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part52.txt +50 -0
  201. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part53.txt +47 -0
  202. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part54.txt +50 -0
  203. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part55.txt +50 -0
  204. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part56.txt +47 -0
  205. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part57.txt +50 -0
  206. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part58.txt +50 -0
  207. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part59.txt +47 -0
  208. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part6.txt +47 -0
  209. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part60.txt +50 -0
  210. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part61.txt +50 -0
  211. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part7.txt +50 -0
  212. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part8.txt +50 -0
  213. rasa/cli/project_templates/finance/docs/huggingface_alpaca_dataset/questions_part9.txt +47 -0
  214. rasa/cli/project_templates/finance/domain/accounts/check_balance.yml +11 -0
  215. rasa/cli/project_templates/finance/domain/cards/block_card.yml +101 -0
  216. rasa/cli/project_templates/finance/domain/cards/select_card.yml +12 -0
  217. rasa/cli/project_templates/finance/domain/general/assistant_details.yml +12 -0
  218. rasa/cli/project_templates/finance/domain/general/bot_identity.yml +5 -0
  219. rasa/cli/project_templates/finance/domain/general/cannot_handle.yml +5 -0
  220. rasa/cli/project_templates/finance/domain/general/defaults.yml +24 -0
  221. rasa/cli/project_templates/finance/domain/general/feedback.yml +28 -0
  222. rasa/cli/project_templates/finance/domain/general/goodbye.yml +7 -0
  223. rasa/cli/project_templates/finance/domain/general/help.yml +5 -0
  224. rasa/cli/project_templates/finance/domain/general/human_handoff.yml +30 -0
  225. rasa/cli/project_templates/finance/domain/general/utils.yml +13 -0
  226. rasa/cli/project_templates/finance/domain/general/welcome.yml +8 -0
  227. rasa/cli/project_templates/finance/domain/transfers/add_payee.yml +47 -0
  228. rasa/cli/project_templates/finance/domain/transfers/list_payees.yml +4 -0
  229. rasa/cli/project_templates/finance/domain/transfers/remove_payee.yml +16 -0
  230. rasa/cli/project_templates/finance/domain/transfers/transfer_money.yml +79 -0
  231. rasa/cli/project_templates/finance/endpoints.yml +63 -0
  232. rasa/cli/project_templates/finance/prompts/rephraser_demo_personality_prompt.jinja2 +19 -0
  233. rasa/cli/project_templates/telco/actions/__init__.py +0 -0
  234. rasa/cli/project_templates/telco/actions/billing/__init__.py +0 -0
  235. rasa/cli/project_templates/telco/actions/billing/actions_billing.py +204 -0
  236. rasa/cli/project_templates/telco/actions/general/__init__.py +0 -0
  237. rasa/cli/project_templates/telco/actions/general/action_human_handoff.py +49 -0
  238. rasa/cli/project_templates/telco/actions/network/__init__.py +0 -0
  239. rasa/cli/project_templates/telco/actions/network/actions_get_data_from_db.py +48 -0
  240. rasa/cli/project_templates/telco/actions/network/actions_run_diagnostics.py +28 -0
  241. rasa/cli/project_templates/telco/actions/network/actions_session_start.py +18 -0
  242. rasa/cli/project_templates/telco/config.yml +27 -0
  243. rasa/cli/project_templates/telco/credentials.yml +33 -0
  244. rasa/cli/project_templates/telco/csvs/billing.csv +19 -0
  245. rasa/cli/project_templates/telco/csvs/customers.csv +5 -0
  246. rasa/cli/project_templates/telco/data/billing/flow_understand_bill.yml +45 -0
  247. rasa/cli/project_templates/telco/data/general/bot_challenge.yml +6 -0
  248. rasa/cli/project_templates/telco/data/general/feedback.yml +20 -0
  249. rasa/cli/project_templates/telco/data/general/goodbye.yml +6 -0
  250. rasa/cli/project_templates/telco/data/general/hello.yml +6 -0
  251. rasa/cli/project_templates/telco/data/general/human_handoff.yml +16 -0
  252. rasa/cli/project_templates/telco/data/general/patterns.yml +30 -0
  253. rasa/cli/project_templates/telco/data/network/flow_reboot_router.yml +8 -0
  254. rasa/cli/project_templates/telco/data/network/flow_reset_router.yml +7 -0
  255. rasa/cli/project_templates/telco/data/network/flow_solve_internet_issue.yml +73 -0
  256. rasa/cli/project_templates/telco/docs/docs.md +5 -0
  257. rasa/cli/project_templates/telco/docs/network/reset_vs_rboot_router.txt +1 -0
  258. rasa/cli/project_templates/telco/docs/network/restart_router.txt +6 -0
  259. rasa/cli/project_templates/telco/docs/network/run_speed_test.txt +6 -0
  260. rasa/cli/project_templates/telco/domain/billing/domain_undertand_bill.yml +102 -0
  261. rasa/cli/project_templates/telco/domain/general/bot_challenge.yml +4 -0
  262. rasa/cli/project_templates/telco/domain/general/feedback.yml +25 -0
  263. rasa/cli/project_templates/telco/domain/general/goodbye.yml +7 -0
  264. rasa/cli/project_templates/telco/domain/general/hello.yml +5 -0
  265. rasa/cli/project_templates/telco/domain/general/human_handoff.yml +29 -0
  266. rasa/cli/project_templates/telco/domain/general/patterns.yml +33 -0
  267. rasa/cli/project_templates/telco/domain/network/domain_reboot_router.yml +21 -0
  268. rasa/cli/project_templates/telco/domain/network/domain_reset_router.yml +12 -0
  269. rasa/cli/project_templates/telco/domain/network/domain_run_speed_test.yml +25 -0
  270. rasa/cli/project_templates/telco/domain/network/domain_solve_internet_issue.yml +75 -0
  271. rasa/cli/project_templates/telco/domain/shared.yml +129 -0
  272. rasa/cli/project_templates/telco/endpoints.yml +63 -0
  273. rasa/cli/project_templates/telco/prompts/rephraser_demo_personality_prompt.jinja2 +40 -0
  274. rasa/cli/project_templates/tutorial/config.yml +2 -1
  275. rasa/cli/scaffold.py +46 -2
  276. rasa/core/actions/action.py +0 -1
  277. rasa/core/channels/channel.py +4 -3
  278. rasa/core/channels/constants.py +3 -0
  279. rasa/core/channels/development_inspector.py +1 -1
  280. rasa/core/channels/inspector/dist/assets/{arc-1ddec37b.js → arc-18042c22.js} +1 -1
  281. rasa/core/channels/inspector/dist/assets/{blockDiagram-38ab4fdb-18af387c.js → blockDiagram-38ab4fdb-fdd6bcfa.js} +1 -1
  282. rasa/core/channels/inspector/dist/assets/{c4Diagram-3d4e48cf-250127a3.js → c4Diagram-3d4e48cf-f5ae6786.js} +1 -1
  283. rasa/core/channels/inspector/dist/assets/channel-b9b536fc.js +1 -0
  284. rasa/core/channels/inspector/dist/assets/{classDiagram-70f12bd4-c3388b34.js → classDiagram-70f12bd4-81efba3e.js} +1 -1
  285. rasa/core/channels/inspector/dist/assets/{classDiagram-v2-f2320105-9c893a82.js → classDiagram-v2-f2320105-3b6b6a92.js} +1 -1
  286. rasa/core/channels/inspector/dist/assets/clone-78d2ddcf.js +1 -0
  287. rasa/core/channels/inspector/dist/assets/{createText-2e5e7dd3-c111213b.js → createText-2e5e7dd3-31422447.js} +1 -1
  288. rasa/core/channels/inspector/dist/assets/{edges-e0da2a9e-812a729d.js → edges-e0da2a9e-518a90db.js} +1 -1
  289. rasa/core/channels/inspector/dist/assets/{erDiagram-9861fffd-fd5051bc.js → erDiagram-9861fffd-a6d3c25a.js} +1 -1
  290. rasa/core/channels/inspector/dist/assets/{flowDb-956e92f1-3287ac02.js → flowDb-956e92f1-e048c2be.js} +1 -1
  291. rasa/core/channels/inspector/dist/assets/{flowDiagram-66a62f08-692fb0b2.js → flowDiagram-66a62f08-c7474c91.js} +1 -1
  292. rasa/core/channels/inspector/dist/assets/flowDiagram-v2-96b9c2cf-8b09c060.js +1 -0
  293. rasa/core/channels/inspector/dist/assets/{flowchart-elk-definition-4a651766-008376f1.js → flowchart-elk-definition-4a651766-cb4d8723.js} +1 -1
  294. rasa/core/channels/inspector/dist/assets/{ganttDiagram-c361ad54-df330a69.js → ganttDiagram-c361ad54-346636a2.js} +1 -1
  295. rasa/core/channels/inspector/dist/assets/{gitGraphDiagram-72cf32ee-e03676fb.js → gitGraphDiagram-72cf32ee-7c508874.js} +1 -1
  296. rasa/core/channels/inspector/dist/assets/{graph-46fad2ba.js → graph-14702d8a.js} +1 -1
  297. rasa/core/channels/inspector/dist/assets/{index-3862675e-a484ac55.js → index-3862675e-f18b534b.js} +1 -1
  298. rasa/core/channels/inspector/dist/assets/{index-a003633f.js → index-4d4bdf3a.js} +231 -231
  299. rasa/core/channels/inspector/dist/assets/{infoDiagram-f8f76790-3f9e6ec2.js → infoDiagram-f8f76790-64154b83.js} +1 -1
  300. rasa/core/channels/inspector/dist/assets/{journeyDiagram-49397b02-79f72383.js → journeyDiagram-49397b02-833a5f95.js} +1 -1
  301. rasa/core/channels/inspector/dist/assets/{layout-aad098e5.js → layout-5a3b2123.js} +1 -1
  302. rasa/core/channels/inspector/dist/assets/{line-219ab7ae.js → line-2272a8c7.js} +1 -1
  303. rasa/core/channels/inspector/dist/assets/{linear-2cddbe62.js → linear-35bcf273.js} +1 -1
  304. rasa/core/channels/inspector/dist/assets/{mindmap-definition-fc14e90a-1d41ed99.js → mindmap-definition-fc14e90a-92dcb0e9.js} +1 -1
  305. rasa/core/channels/inspector/dist/assets/{pieDiagram-8a3498a8-cc496ee8.js → pieDiagram-8a3498a8-94dbc900.js} +1 -1
  306. rasa/core/channels/inspector/dist/assets/{quadrantDiagram-120e2f19-84d32884.js → quadrantDiagram-120e2f19-8b7a9c33.js} +1 -1
  307. rasa/core/channels/inspector/dist/assets/{requirementDiagram-deff3bca-c0deb984.js → requirementDiagram-deff3bca-6f7eab81.js} +1 -1
  308. rasa/core/channels/inspector/dist/assets/{sankeyDiagram-04a897e0-b9d7fd62.js → sankeyDiagram-04a897e0-f43e581d.js} +1 -1
  309. rasa/core/channels/inspector/dist/assets/{sequenceDiagram-704730f1-7d517565.js → sequenceDiagram-704730f1-0bcbefc3.js} +1 -1
  310. rasa/core/channels/inspector/dist/assets/{stateDiagram-587899a1-98ef9b27.js → stateDiagram-587899a1-b8a74083.js} +1 -1
  311. rasa/core/channels/inspector/dist/assets/{stateDiagram-v2-d93cdb3a-cee70748.js → stateDiagram-v2-d93cdb3a-2070218f.js} +1 -1
  312. rasa/core/channels/inspector/dist/assets/{styles-6aaf32cf-3f9d1c96.js → styles-6aaf32cf-f1d54e34.js} +1 -1
  313. rasa/core/channels/inspector/dist/assets/{styles-9a916d00-67471923.js → styles-9a916d00-980de489.js} +1 -1
  314. rasa/core/channels/inspector/dist/assets/{styles-c10674c1-bd093fb7.js → styles-c10674c1-3c03abde.js} +1 -1
  315. rasa/core/channels/inspector/dist/assets/{svgDrawCommon-08f97a94-675794e8.js → svgDrawCommon-08f97a94-46ba068f.js} +1 -1
  316. rasa/core/channels/inspector/dist/assets/{timeline-definition-85554ec2-0ac67617.js → timeline-definition-85554ec2-901f5e3d.js} +1 -1
  317. rasa/core/channels/inspector/dist/assets/{xychartDiagram-e933f94c-c018dc37.js → xychartDiagram-e933f94c-acbc628a.js} +1 -1
  318. rasa/core/channels/inspector/dist/index.html +2 -2
  319. rasa/core/channels/inspector/index.html +1 -1
  320. rasa/core/channels/inspector/src/App.tsx +10 -11
  321. rasa/core/channels/inspector/src/components/DialogueInformation.tsx +12 -3
  322. rasa/core/channels/socketio.py +212 -51
  323. rasa/core/channels/studio_chat.py +43 -23
  324. rasa/core/channels/voice_stream/voice_channel.py +5 -3
  325. rasa/core/policies/enterprise_search_policy.py +4 -7
  326. rasa/core/policies/flows/flow_executor.py +8 -1
  327. rasa/core/run.py +13 -3
  328. rasa/dialogue_understanding/generator/flow_retrieval.py +10 -9
  329. rasa/dialogue_understanding/patterns/default_flows_for_patterns.yml +1 -1
  330. rasa/engine/storage/local_model_storage.py +45 -2
  331. rasa/model_manager/model_api.py +4 -5
  332. rasa/model_manager/runner_service.py +1 -1
  333. rasa/model_manager/socket_bridge.py +20 -14
  334. rasa/model_manager/trainer_service.py +12 -9
  335. rasa/model_manager/utils.py +1 -29
  336. rasa/shared/core/domain.py +62 -15
  337. rasa/shared/core/flows/flow_step.py +7 -1
  338. rasa/shared/core/flows/steps/call.py +8 -1
  339. rasa/shared/core/flows/yaml_flows_io.py +16 -8
  340. rasa/shared/core/slots.py +4 -0
  341. rasa/shared/importers/importer.py +6 -0
  342. rasa/shared/importers/utils.py +77 -1
  343. rasa/shared/providers/_utils.py +60 -44
  344. rasa/shared/providers/embedding/default_litellm_embedding_client.py +2 -0
  345. rasa/shared/providers/llm/default_litellm_llm_client.py +2 -0
  346. rasa/studio/upload.py +12 -46
  347. rasa/telemetry.py +97 -23
  348. rasa/utils/io.py +27 -9
  349. rasa/utils/json_utils.py +6 -1
  350. rasa/utils/log_utils.py +5 -1
  351. rasa/utils/openapi.py +144 -0
  352. rasa/validator.py +7 -3
  353. rasa/version.py +1 -1
  354. {rasa_pro-3.14.0.dev20250818.dist-info → rasa_pro-3.14.0.dev20250901.dist-info}/METADATA +9 -10
  355. {rasa_pro-3.14.0.dev20250818.dist-info → rasa_pro-3.14.0.dev20250901.dist-info}/RECORD +358 -83
  356. rasa/core/channels/inspector/dist/assets/channel-59f6d54b.js +0 -1
  357. rasa/core/channels/inspector/dist/assets/clone-26177ddb.js +0 -1
  358. rasa/core/channels/inspector/dist/assets/flowDiagram-v2-96b9c2cf-29c03f5a.js +0 -1
  359. {rasa_pro-3.14.0.dev20250818.dist-info → rasa_pro-3.14.0.dev20250901.dist-info}/NOTICE +0 -0
  360. {rasa_pro-3.14.0.dev20250818.dist-info → rasa_pro-3.14.0.dev20250901.dist-info}/WHEEL +0 -0
  361. {rasa_pro-3.14.0.dev20250818.dist-info → rasa_pro-3.14.0.dev20250901.dist-info}/entry_points.txt +0 -0
@@ -0,0 +1,206 @@
1
+ """Lakera AI guardrails provider implementation."""
2
+
3
+ import asyncio
4
+ import os
5
+ import time
6
+ from contextlib import asynccontextmanager
7
+ from typing import Any, AsyncGenerator, Dict, Optional
8
+
9
+ import aiohttp
10
+ import structlog
11
+
12
+ from rasa.builder import config
13
+ from rasa.builder.guardrails.constants import (
14
+ LAKERA_API_KEY_ENV_VAR,
15
+ LAKERA_GUARD_ENDPOINT,
16
+ LAKERA_GUARD_RESULTS_ENDPOINT,
17
+ )
18
+ from rasa.builder.guardrails.exceptions import GuardrailsError
19
+ from rasa.builder.guardrails.models import (
20
+ GuardrailResponse,
21
+ LakeraGuardrailRequest,
22
+ LakeraGuardrailResponse,
23
+ )
24
+
25
+ structlogger = structlog.get_logger()
26
+
27
+
28
+ class LakeraAIGuardrails:
29
+ """Guardrails provider using Lakera AI."""
30
+
31
+ def __init__(
32
+ self,
33
+ api_key: Optional[str] = None,
34
+ base_url: Optional[str] = None,
35
+ ):
36
+ """Initialize Lakera guardrails provider.
37
+
38
+ Args:
39
+ severity_threshold: The minimum severity level for the guardrail to flag
40
+ the user request as unsafe.
41
+ api_key: Lakera AI API key.
42
+ base_url: Optional base URL for the API. If not provided, the default
43
+ Lakera API URL (https://api.lakera.ai/v2) will be used.
44
+ """
45
+ self.base_url: str = base_url or config.LAKERA_BASE_URL
46
+ self._api_key: Optional[str] = api_key or os.getenv(LAKERA_API_KEY_ENV_VAR)
47
+ self._session: Optional[aiohttp.ClientSession] = None
48
+
49
+ @property
50
+ def guard_endpoint(self) -> str:
51
+ """Get the guard endpoint for the Lakera API."""
52
+ return f"{self.base_url}/{LAKERA_GUARD_ENDPOINT}"
53
+
54
+ @property
55
+ def guard_results_endpoint(self) -> str:
56
+ """Get the guard results endpoint for the Lakera API."""
57
+ return f"{self.base_url}/{LAKERA_GUARD_RESULTS_ENDPOINT}"
58
+
59
+ @asynccontextmanager
60
+ async def _get_session(self) -> AsyncGenerator[aiohttp.ClientSession, None]:
61
+ """Create a fresh ClientSession, yield it, and always close it."""
62
+ session = aiohttp.ClientSession(headers=self._get_headers())
63
+ structlogger.debug("lakera_guardrails._get_session", base_url=self.base_url)
64
+
65
+ try:
66
+ yield session
67
+ except Exception as e:
68
+ structlogger.error("lakera_guardrails.session_error", error=str(e))
69
+ raise
70
+ finally:
71
+ try:
72
+ await session.close()
73
+ except Exception as exc:
74
+ structlogger.warning(
75
+ "lakera_guardrails.session_close_error",
76
+ event_info="Failed to close aiohttp client session cleanly.",
77
+ error=str(exc),
78
+ )
79
+
80
+ def _get_headers(self) -> Dict[str, str]:
81
+ """Get the headers for the Lakera API request.
82
+
83
+ Returns:
84
+ A dictionary containing the Authorization header with the API key.
85
+ """
86
+ using_proxy = bool(config.HELLO_LLM_PROXY_BASE_URL)
87
+
88
+ if using_proxy:
89
+ if not config.RASA_PRO_LICENSE:
90
+ raise GuardrailsError(
91
+ "HELLO_LLM_PROXY_BASE_URL is set but RASA_PRO_LICENSE is missing. "
92
+ "Proxy requires a Rasa Pro license token for authentication."
93
+ )
94
+ return {"Authorization": f"Bearer {config.RASA_PRO_LICENSE}"}
95
+
96
+ if not self._api_key:
97
+ raise GuardrailsError(
98
+ "LAKERA_API_KEY is missing. Provide it via env LAKERA_API_KEY or "
99
+ "pass api_key= to LakeraAIGuardrails."
100
+ )
101
+ return {"Authorization": f"Bearer {self._api_key}"}
102
+
103
+ async def send_request(self, request: LakeraGuardrailRequest) -> GuardrailResponse:
104
+ """Send a request to the Lakera API.
105
+
106
+ Args:
107
+ request: The guardrail request to send to the Lakera API.
108
+
109
+ Returns:
110
+ GuardrailResponse with the results of the check.
111
+
112
+ Raises:
113
+ GuardrailsError: If the request times out or returns a non-200 status code.
114
+ Exception: If the request fails for any other reason.
115
+ """
116
+ start_time = time.time()
117
+ try:
118
+ async with self._get_session() as session:
119
+ raw_response = await self._send_http_request(session, request)
120
+ response = LakeraGuardrailResponse.from_raw_response(
121
+ raw_response,
122
+ hello_rasa_user_id=request.hello_rasa_user_id,
123
+ hello_rasa_project_id=request.hello_rasa_project_id,
124
+ )
125
+ processing_time_ms = (time.time() - start_time) * 1000
126
+ response.processing_time_ms = processing_time_ms
127
+ return response
128
+
129
+ # Propagate the GuardrailsError if it occurs.
130
+ except GuardrailsError as e:
131
+ raise e
132
+
133
+ except asyncio.TimeoutError:
134
+ message = "Lakera API request timed out."
135
+ structlogger.error(
136
+ "lakera_guardrails.send_request.timeout_error",
137
+ event_info=message,
138
+ processing_time_ms=(time.time() - start_time) * 1000,
139
+ )
140
+ raise GuardrailsError(message)
141
+
142
+ # Propagate the unexpected exceptions.
143
+ except Exception as e:
144
+ message = "Lakera API request failed."
145
+ structlogger.error(
146
+ "lakera_guardrails.send_request.unexpected_error",
147
+ event_info="Lakera API request failed.",
148
+ request=request,
149
+ error=e,
150
+ processing_time_ms=(time.time() - start_time) * 1000,
151
+ )
152
+ raise e
153
+
154
+ async def _send_http_request(
155
+ self, session: aiohttp.ClientSession, request: LakeraGuardrailRequest
156
+ ) -> Dict[str, Any]:
157
+ """Make an HTTP request to the Lakera API.
158
+
159
+ Args:
160
+ session: The aiohttp session to use for the request.
161
+ request: The guardrail request to send.
162
+
163
+ Returns:
164
+ The raw JSON response from the API.
165
+
166
+ Raises:
167
+ GuardrailsError: If the request fails or response parsing fails.
168
+ """
169
+ # Log the request details for debugging
170
+ json_payload = request.to_json_payload()
171
+ structlogger.debug(
172
+ "lakera_guardrails.send_request.request",
173
+ url=self.guard_endpoint,
174
+ method="POST",
175
+ request_body=json_payload,
176
+ )
177
+
178
+ async with session.post(
179
+ self.guard_endpoint,
180
+ json=json_payload,
181
+ ) as client_response:
182
+ # Check if the response is successful. If not, raise an error.
183
+ if client_response.status >= 400:
184
+ error_text = await client_response.text()
185
+ message = (
186
+ f"Lakera API request failed with status "
187
+ f"`{client_response.status}`. Error: "
188
+ f"`{error_text}`."
189
+ )
190
+ structlogger.error(
191
+ "lakera_guardrails.send_request.http_error",
192
+ event_info=message,
193
+ url=self.guard_endpoint,
194
+ status=client_response.status,
195
+ error=error_text,
196
+ request_body=json_payload,
197
+ )
198
+ raise GuardrailsError(message)
199
+
200
+ # Parse the response as a dictionary.
201
+ raw_response = await client_response.json()
202
+ structlogger.debug(
203
+ "lakera_guardrails.send_request.response",
204
+ response_body=raw_response,
205
+ )
206
+ return raw_response
@@ -0,0 +1,231 @@
1
+ """Models for guardrails system."""
2
+
3
+ import copy
4
+ from abc import ABC, abstractmethod
5
+ from enum import Enum
6
+ from typing import Any, Dict, List, Optional
7
+
8
+ from pydantic import BaseModel, ConfigDict, Field
9
+
10
+
11
+ class GuardrailType(Enum):
12
+ """Types of guardrails that can be applied with Lakera AI."""
13
+
14
+ PROMPT_ATTACK = "prompt_attack"
15
+ CONTENT_VIOLATION = "content_violation"
16
+ DATA_LEAKAGE = "data_leakage"
17
+ MALICIOUS_LINKS = "malicious_content"
18
+ CUSTOM = "custom"
19
+ OTHER = "other"
20
+
21
+
22
+ class GuardrailRequest(BaseModel, ABC):
23
+ """Request for guardrails check."""
24
+
25
+ hello_rasa_user_id: Optional[str] = Field(
26
+ default=None,
27
+ description="Required. User identifier for the Hello Rasa project. ",
28
+ )
29
+ hello_rasa_project_id: Optional[str] = Field(
30
+ default=None,
31
+ description="Required. Project identifier for the Hello Rasa project. ",
32
+ )
33
+ metadata: Optional[Dict[str, Any]] = Field(
34
+ default=None, description="Additional metadata for the guardrails endpoint."
35
+ )
36
+
37
+ @abstractmethod
38
+ def to_json_payload(self) -> Dict[str, Any]:
39
+ """Convert the request to a JSON payload."""
40
+ ...
41
+
42
+
43
+ class LakeraGuardrailRequest(GuardrailRequest):
44
+ """Request for Lakera guardrails check."""
45
+
46
+ lakera_project_id: str = Field(
47
+ description="Required. Project identifier for the Lakera AI project."
48
+ )
49
+ payload: bool = Field(
50
+ default=True,
51
+ description=(
52
+ "From Lakera AI: When true the response will return a payload object "
53
+ "containing any PII, profanity or custom detector regex matches detected, "
54
+ "along with their location within the contents."
55
+ ),
56
+ )
57
+ breakdown: bool = Field(
58
+ default=True,
59
+ description=(
60
+ "From Lakera AI: When true the response will return a breakdown list of "
61
+ "the detectors that were run, as defined in the policy, and whether each "
62
+ "of them detected something or not."
63
+ ),
64
+ )
65
+
66
+ messages: List[Dict[str, Any]] = Field(
67
+ description=(
68
+ "Required. From Lakera AI: List of messages comprising the interaction "
69
+ "history with the LLM in OpenAI API Chat Completions format. Can be "
70
+ "multiple messages of any role: user, assistant, system, tool, or "
71
+ "developer."
72
+ ),
73
+ )
74
+
75
+ def to_json_payload(self) -> Dict[str, Any]:
76
+ """Convert the request to a JSON payload to be sent to the Lakera endpoint."""
77
+ metadata = self.metadata or {}
78
+ metadata["hello_rasa_project_id"] = self.hello_rasa_project_id
79
+ metadata["hello_rasa_user_id"] = self.hello_rasa_user_id
80
+
81
+ json_payload: Dict[str, Any] = {
82
+ "messages": self.messages,
83
+ "project_id": self.lakera_project_id,
84
+ "metadata": metadata,
85
+ }
86
+
87
+ if self.payload:
88
+ json_payload["payload"] = self.payload
89
+ if self.breakdown:
90
+ json_payload["breakdown"] = self.breakdown
91
+
92
+ return json_payload
93
+
94
+
95
+ class GuardrailDetection(BaseModel):
96
+ """Represents a single guardrail detection."""
97
+
98
+ type: GuardrailType = Field(description="Type of guardrail detection.")
99
+ original_type: str = Field(description="Original detection from the provider.")
100
+ metadata: Optional[Dict[str, Any]] = Field(
101
+ default=None, description="Additional metadata about the detection itself."
102
+ )
103
+
104
+
105
+ class GuardrailResponse(BaseModel):
106
+ """Response from guardrails system."""
107
+
108
+ hello_rasa_user_id: Optional[str] = Field(
109
+ default=None,
110
+ description="Required. User identifier for the Hello Rasa project. ",
111
+ )
112
+ hello_rasa_project_id: Optional[str] = Field(
113
+ default=None,
114
+ description="Required. Project identifier for the Hello Rasa project. ",
115
+ )
116
+ flagged: bool = Field(description="Whether any policy violations were detected.")
117
+ detections: List[GuardrailDetection] = Field(
118
+ default_factory=list, description="List of detected policy violations."
119
+ )
120
+ processing_time_ms: Optional[float] = Field(
121
+ default=None, description="Processing time in milliseconds."
122
+ )
123
+ metadata: Optional[Dict[str, Any]] = Field(
124
+ default=None, description="Additional metadata from the provider."
125
+ )
126
+
127
+
128
+ class LakeraGuardrailResponse(GuardrailResponse):
129
+ """Response from Lakera AI `/guard` endpoint."""
130
+
131
+ @classmethod
132
+ def from_raw_response(
133
+ cls,
134
+ raw_response: Dict[str, Any],
135
+ hello_rasa_user_id: str,
136
+ hello_rasa_project_id: str,
137
+ ) -> "LakeraGuardrailResponse":
138
+ """Create a LakeraGuardrailResponse from a response."""
139
+ from rasa.builder.guardrails.utils import (
140
+ map_lakera_detector_type_to_guardrail_type,
141
+ )
142
+
143
+ # Get the basic information from the response and create the response object.
144
+ flagged = raw_response.get("flagged", False)
145
+ metadata = raw_response.get("metadata")
146
+ response = cls(
147
+ flagged=flagged,
148
+ metadata=metadata,
149
+ hello_rasa_user_id=hello_rasa_user_id,
150
+ hello_rasa_project_id=hello_rasa_project_id,
151
+ )
152
+
153
+ # If the response is not flagged, return the response object.
154
+ if not flagged:
155
+ return response
156
+
157
+ # If the response is flagged, parse the breakdown section.
158
+ breakdown = raw_response.get("breakdown", [])
159
+
160
+ # Parse the breakdown.
161
+ detections: List[GuardrailDetection] = []
162
+ for detector in breakdown:
163
+ if detector.get("detected", True):
164
+ detector_type = detector.get("detector_type")
165
+ rasa_detection_type = map_lakera_detector_type_to_guardrail_type(
166
+ detector_type,
167
+ )
168
+ if not rasa_detection_type:
169
+ continue
170
+ # Remove the detector type and the detected flag from the detector.
171
+ # And keep the rest as part of the metadata. In case of a Lakera this
172
+ # will include the message_id, project_id, policy_id, detector_id, etc.
173
+ metadata = copy.deepcopy(detector)
174
+ metadata.pop("detected")
175
+ metadata.pop("detector_type")
176
+
177
+ detections.append(
178
+ GuardrailDetection(
179
+ type=rasa_detection_type,
180
+ original_type=detector_type,
181
+ metadata=metadata,
182
+ )
183
+ )
184
+
185
+ # If there are detections, add them to the response.
186
+ if detections:
187
+ response.detections = detections
188
+
189
+ return response
190
+
191
+
192
+ class GuardrailRequestKey(BaseModel):
193
+ user_text: str
194
+ hello_rasa_user_id: str = ""
195
+ hello_rasa_project_id: str = ""
196
+ lakera_project_id: str
197
+
198
+ # hashable by value
199
+ model_config = ConfigDict(frozen=True)
200
+
201
+
202
+ class ScopeState(BaseModel):
203
+ blocked_until: Optional[float] = Field(
204
+ default=None,
205
+ description="UNIX timestamp in seconds until this scope is blocked.",
206
+ )
207
+ violations: List[float] = Field(
208
+ default_factory=list,
209
+ description="UNIX timestamps in seconds when violations occurred.",
210
+ )
211
+
212
+ def is_blocked(self, now: float) -> bool:
213
+ """Return whether the scope is currently blocked.
214
+
215
+ Args:
216
+ now: Current time as a UNIX timestamp.
217
+
218
+ Returns:
219
+ True if blocked_until is set and in the future, else False.
220
+ """
221
+ return self.blocked_until is not None and now < self.blocked_until
222
+
223
+
224
+ class ProjectState(BaseModel):
225
+ project: ScopeState = Field(default_factory=ScopeState)
226
+ users: Dict[str, ScopeState] = Field(default_factory=dict)
227
+
228
+
229
+ class BlockResult(BaseModel):
230
+ user_blocked_now: bool = False
231
+ project_blocked_now: bool = False
@@ -0,0 +1,238 @@
1
+ import asyncio
2
+ import time
3
+ from typing import Optional
4
+
5
+ import structlog
6
+
7
+ from rasa.builder import config
8
+ from rasa.builder.guardrails.constants import (
9
+ BLOCK_SCOPE_PROJECT,
10
+ BLOCK_SCOPE_USER,
11
+ BlockScope,
12
+ )
13
+ from rasa.builder.guardrails.models import BlockResult, ProjectState, ScopeState
14
+
15
+ structlogger = structlog.get_logger()
16
+
17
+
18
+ class GuardrailsInMemoryStore:
19
+ """In-memory strike and block tracking for a single project instance."""
20
+
21
+ def __init__(self) -> None:
22
+ self._state: ProjectState = ProjectState()
23
+ self._lock = asyncio.Lock()
24
+
25
+ @staticmethod
26
+ def _normalize_user_id(user_id: Optional[str]) -> str:
27
+ """Normalize and validate user_id.
28
+
29
+ Args:
30
+ user_id: User identifier.
31
+
32
+ Returns:
33
+ Validated, non-empty user id.
34
+ """
35
+ uid = (user_id or "").strip()
36
+ if not uid:
37
+ raise ValueError("user_id is required for guardrails tracking.")
38
+ return uid
39
+
40
+ def _get_or_create_user(self, user_id: str) -> ScopeState:
41
+ """Return (and create if needed) the per-user scope state.
42
+
43
+ Args:
44
+ user_id: User identifier.
45
+
46
+ Returns:
47
+ Mutable ScopeState for the user.
48
+ """
49
+ uid = self._normalize_user_id(user_id)
50
+ return self._state.users.setdefault(uid, ScopeState())
51
+
52
+ def _clear_expired_project_block(self, now: float) -> None:
53
+ """Clear project-level block if expired.
54
+
55
+ Args:
56
+ now: Current time as a UNIX timestamp.
57
+ """
58
+ project = self._state.project
59
+ if project.blocked_until and now >= project.blocked_until:
60
+ project.blocked_until = None
61
+
62
+ def _clear_expired_user_block(self, user_id: str, now: float) -> None:
63
+ """Clear user-level block if expired.
64
+
65
+ Args:
66
+ user_id: User identifier.
67
+ now: Current time as a UNIX timestamp.
68
+ """
69
+ try:
70
+ uid = self._normalize_user_id(user_id)
71
+ except ValueError:
72
+ return
73
+
74
+ user_state = self._state.users.get(uid)
75
+ if user_state and user_state.blocked_until and now >= user_state.blocked_until:
76
+ user_state.blocked_until = None
77
+
78
+ @staticmethod
79
+ def _apply_user_block_if_needed(user_id: str, user_state: ScopeState) -> bool:
80
+ """Apply a user-level block if the threshold is reached.
81
+
82
+ Args:
83
+ user_id: User identifier.
84
+ user_state: Current user's scope state.
85
+
86
+ Returns:
87
+ True if a block was applied during this call; otherwise False.
88
+ """
89
+ now = time.time()
90
+ if user_state.is_blocked(now):
91
+ return False
92
+
93
+ if len(user_state.violations) < config.GUARDRAILS_USER_MAX_STRIKES:
94
+ return False
95
+
96
+ duration_seconds = config.GUARDRAILS_BLOCK_DURATION_SECONDS
97
+ block_until = float("inf") if duration_seconds <= 0 else now + duration_seconds
98
+ user_state.blocked_until = block_until
99
+ structlogger.info(
100
+ "guardrails.store.user_blocked",
101
+ project_id=config.HELLO_RASA_PROJECT_ID,
102
+ user_id=user_id,
103
+ strikes=len(user_state.violations),
104
+ duration_sec=duration_seconds,
105
+ )
106
+ return True
107
+
108
+ def _apply_project_block_if_needed(self) -> bool:
109
+ """Apply a project-level block if the threshold is reached.
110
+
111
+ Returns:
112
+ True if a block was applied during this call; otherwise False.
113
+ """
114
+ now = time.time()
115
+ project_state = self._state.project
116
+ if project_state.is_blocked(now):
117
+ return False
118
+
119
+ if len(project_state.violations) < config.GUARDRAILS_PROJECT_MAX_STRIKES:
120
+ return False
121
+
122
+ duration_seconds = config.GUARDRAILS_BLOCK_DURATION_SECONDS
123
+ block_until = float("inf") if duration_seconds <= 0 else now + duration_seconds
124
+ project_state.blocked_until = block_until
125
+ structlogger.info(
126
+ "guardrails.store.project_blocked",
127
+ project_id=config.HELLO_RASA_PROJECT_ID,
128
+ strikes=len(project_state.violations),
129
+ duration_sec=duration_seconds,
130
+ )
131
+ return True
132
+
133
+ async def check_block_scope(self, user_id: str) -> Optional[BlockScope]:
134
+ """Return current block scope if blocked.
135
+
136
+ Args:
137
+ user_id: User identifier.
138
+
139
+ Returns:
140
+ 'user' if user blocked, 'project' if project blocked, otherwise None.
141
+ """
142
+ uid = self._normalize_user_id(user_id)
143
+
144
+ async with self._lock:
145
+ now = time.time()
146
+
147
+ # Clear expired blocks
148
+ self._clear_expired_project_block(now)
149
+ self._clear_expired_user_block(uid, now)
150
+
151
+ user_state = self._state.users.get(uid)
152
+
153
+ if user_state and user_state.is_blocked(now):
154
+ return BLOCK_SCOPE_USER
155
+ if self._state.project.is_blocked(now):
156
+ return BLOCK_SCOPE_PROJECT
157
+ return None
158
+
159
+ async def record_violation(self, user_id: str) -> BlockResult:
160
+ """Record a violation and apply thresholds/blocks.
161
+
162
+ Args:
163
+ user_id: User identifier.
164
+
165
+ Returns:
166
+ BlockResult indicating whether a new user or project block was applied.
167
+ """
168
+ uid = self._normalize_user_id(user_id)
169
+ result = BlockResult()
170
+
171
+ async with self._lock:
172
+ now = time.time()
173
+ user_state = self._get_or_create_user(uid)
174
+
175
+ # Accumulate violations (no sliding window)
176
+ self._state.project.violations.append(now)
177
+ user_state.violations.append(now)
178
+
179
+ # Apply user and project blocks
180
+ result.user_blocked_now = self._apply_user_block_if_needed(
181
+ user_id=uid,
182
+ user_state=user_state,
183
+ )
184
+ result.project_blocked_now = self._apply_project_block_if_needed()
185
+
186
+ return result
187
+
188
+ async def unblock_user(self, user_id: str) -> None:
189
+ """Unblock a user without altering strike history.
190
+
191
+ Args:
192
+ user_id: User identifier.
193
+ """
194
+ uid = self._normalize_user_id(user_id)
195
+
196
+ async with self._lock:
197
+ if state := self._state.users.get(uid):
198
+ state.blocked_until = None
199
+
200
+ async def reset_user(self, user_id: str) -> None:
201
+ """Unblock a user and clear their strikes.
202
+
203
+ Args:
204
+ user_id: User identifier.
205
+ """
206
+ uid = self._normalize_user_id(user_id)
207
+
208
+ async with self._lock:
209
+ if state := self._state.users.get(uid):
210
+ state.blocked_until = None
211
+ state.violations.clear()
212
+
213
+ async def unblock_project(self) -> None:
214
+ """Unblock the project without altering strike history."""
215
+ async with self._lock:
216
+ self._state.project.blocked_until = None
217
+
218
+ async def reset_project(self) -> None:
219
+ """Unblock the project and clear project-wide strikes."""
220
+ async with self._lock:
221
+ self._state.project.blocked_until = None
222
+ self._state.project.violations.clear()
223
+
224
+ async def reset_all(self) -> None:
225
+ """Reset all guardrail state (project and users)."""
226
+ async with self._lock:
227
+ # Clear project scope
228
+ self._state.project.blocked_until = None
229
+ self._state.project.violations.clear()
230
+
231
+ # Clear all users
232
+ for state in self._state.users.values():
233
+ state.blocked_until = None
234
+ state.violations.clear()
235
+
236
+
237
+ # Singleton instance for application-wide use
238
+ guardrails_store = GuardrailsInMemoryStore()