rasa-pro 3.11.3__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 (779) hide show
  1. README.md +41 -0
  2. rasa/__init__.py +9 -0
  3. rasa/__main__.py +177 -0
  4. rasa/anonymization/__init__.py +2 -0
  5. rasa/anonymization/anonymisation_rule_yaml_reader.py +91 -0
  6. rasa/anonymization/anonymization_pipeline.py +286 -0
  7. rasa/anonymization/anonymization_rule_executor.py +260 -0
  8. rasa/anonymization/anonymization_rule_orchestrator.py +120 -0
  9. rasa/anonymization/schemas/config.yml +47 -0
  10. rasa/anonymization/utils.py +118 -0
  11. rasa/api.py +160 -0
  12. rasa/cli/__init__.py +5 -0
  13. rasa/cli/arguments/__init__.py +0 -0
  14. rasa/cli/arguments/data.py +106 -0
  15. rasa/cli/arguments/default_arguments.py +207 -0
  16. rasa/cli/arguments/evaluate.py +65 -0
  17. rasa/cli/arguments/export.py +51 -0
  18. rasa/cli/arguments/interactive.py +74 -0
  19. rasa/cli/arguments/run.py +219 -0
  20. rasa/cli/arguments/shell.py +17 -0
  21. rasa/cli/arguments/test.py +211 -0
  22. rasa/cli/arguments/train.py +279 -0
  23. rasa/cli/arguments/visualize.py +34 -0
  24. rasa/cli/arguments/x.py +30 -0
  25. rasa/cli/data.py +354 -0
  26. rasa/cli/e2e_test.py +259 -0
  27. rasa/cli/evaluate.py +222 -0
  28. rasa/cli/export.py +250 -0
  29. rasa/cli/inspect.py +75 -0
  30. rasa/cli/interactive.py +166 -0
  31. rasa/cli/license.py +65 -0
  32. rasa/cli/llm_fine_tuning.py +403 -0
  33. rasa/cli/markers.py +78 -0
  34. rasa/cli/project_templates/__init__.py +0 -0
  35. rasa/cli/project_templates/calm/actions/__init__.py +0 -0
  36. rasa/cli/project_templates/calm/actions/action_template.py +27 -0
  37. rasa/cli/project_templates/calm/actions/add_contact.py +30 -0
  38. rasa/cli/project_templates/calm/actions/db.py +57 -0
  39. rasa/cli/project_templates/calm/actions/list_contacts.py +22 -0
  40. rasa/cli/project_templates/calm/actions/remove_contact.py +35 -0
  41. rasa/cli/project_templates/calm/config.yml +10 -0
  42. rasa/cli/project_templates/calm/credentials.yml +33 -0
  43. rasa/cli/project_templates/calm/data/flows/add_contact.yml +31 -0
  44. rasa/cli/project_templates/calm/data/flows/list_contacts.yml +14 -0
  45. rasa/cli/project_templates/calm/data/flows/remove_contact.yml +29 -0
  46. rasa/cli/project_templates/calm/db/contacts.json +10 -0
  47. rasa/cli/project_templates/calm/domain/add_contact.yml +39 -0
  48. rasa/cli/project_templates/calm/domain/list_contacts.yml +17 -0
  49. rasa/cli/project_templates/calm/domain/remove_contact.yml +38 -0
  50. rasa/cli/project_templates/calm/domain/shared.yml +10 -0
  51. rasa/cli/project_templates/calm/e2e_tests/cancelations/user_cancels_during_a_correction.yml +16 -0
  52. rasa/cli/project_templates/calm/e2e_tests/cancelations/user_changes_mind_on_a_whim.yml +7 -0
  53. rasa/cli/project_templates/calm/e2e_tests/corrections/user_corrects_contact_handle.yml +20 -0
  54. rasa/cli/project_templates/calm/e2e_tests/corrections/user_corrects_contact_name.yml +19 -0
  55. rasa/cli/project_templates/calm/e2e_tests/happy_paths/user_adds_contact_to_their_list.yml +15 -0
  56. rasa/cli/project_templates/calm/e2e_tests/happy_paths/user_lists_contacts.yml +5 -0
  57. rasa/cli/project_templates/calm/e2e_tests/happy_paths/user_removes_contact.yml +11 -0
  58. rasa/cli/project_templates/calm/e2e_tests/happy_paths/user_removes_contact_from_list.yml +12 -0
  59. rasa/cli/project_templates/calm/endpoints.yml +58 -0
  60. rasa/cli/project_templates/default/actions/__init__.py +0 -0
  61. rasa/cli/project_templates/default/actions/actions.py +27 -0
  62. rasa/cli/project_templates/default/config.yml +44 -0
  63. rasa/cli/project_templates/default/credentials.yml +33 -0
  64. rasa/cli/project_templates/default/data/nlu.yml +91 -0
  65. rasa/cli/project_templates/default/data/rules.yml +13 -0
  66. rasa/cli/project_templates/default/data/stories.yml +30 -0
  67. rasa/cli/project_templates/default/domain.yml +34 -0
  68. rasa/cli/project_templates/default/endpoints.yml +42 -0
  69. rasa/cli/project_templates/default/tests/test_stories.yml +91 -0
  70. rasa/cli/project_templates/tutorial/actions/__init__.py +0 -0
  71. rasa/cli/project_templates/tutorial/actions/actions.py +22 -0
  72. rasa/cli/project_templates/tutorial/config.yml +12 -0
  73. rasa/cli/project_templates/tutorial/credentials.yml +33 -0
  74. rasa/cli/project_templates/tutorial/data/flows.yml +8 -0
  75. rasa/cli/project_templates/tutorial/data/patterns.yml +11 -0
  76. rasa/cli/project_templates/tutorial/domain.yml +35 -0
  77. rasa/cli/project_templates/tutorial/endpoints.yml +55 -0
  78. rasa/cli/run.py +143 -0
  79. rasa/cli/scaffold.py +273 -0
  80. rasa/cli/shell.py +141 -0
  81. rasa/cli/studio/__init__.py +0 -0
  82. rasa/cli/studio/download.py +62 -0
  83. rasa/cli/studio/studio.py +296 -0
  84. rasa/cli/studio/train.py +59 -0
  85. rasa/cli/studio/upload.py +62 -0
  86. rasa/cli/telemetry.py +102 -0
  87. rasa/cli/test.py +280 -0
  88. rasa/cli/train.py +278 -0
  89. rasa/cli/utils.py +484 -0
  90. rasa/cli/visualize.py +40 -0
  91. rasa/cli/x.py +206 -0
  92. rasa/constants.py +45 -0
  93. rasa/core/__init__.py +17 -0
  94. rasa/core/actions/__init__.py +0 -0
  95. rasa/core/actions/action.py +1318 -0
  96. rasa/core/actions/action_clean_stack.py +59 -0
  97. rasa/core/actions/action_exceptions.py +24 -0
  98. rasa/core/actions/action_hangup.py +29 -0
  99. rasa/core/actions/action_repeat_bot_messages.py +89 -0
  100. rasa/core/actions/action_run_slot_rejections.py +210 -0
  101. rasa/core/actions/action_trigger_chitchat.py +31 -0
  102. rasa/core/actions/action_trigger_flow.py +109 -0
  103. rasa/core/actions/action_trigger_search.py +31 -0
  104. rasa/core/actions/constants.py +5 -0
  105. rasa/core/actions/custom_action_executor.py +191 -0
  106. rasa/core/actions/direct_custom_actions_executor.py +80 -0
  107. rasa/core/actions/e2e_stub_custom_action_executor.py +72 -0
  108. rasa/core/actions/forms.py +741 -0
  109. rasa/core/actions/grpc_custom_action_executor.py +251 -0
  110. rasa/core/actions/http_custom_action_executor.py +145 -0
  111. rasa/core/actions/loops.py +114 -0
  112. rasa/core/actions/two_stage_fallback.py +186 -0
  113. rasa/core/agent.py +559 -0
  114. rasa/core/auth_retry_tracker_store.py +122 -0
  115. rasa/core/brokers/__init__.py +0 -0
  116. rasa/core/brokers/broker.py +126 -0
  117. rasa/core/brokers/file.py +58 -0
  118. rasa/core/brokers/kafka.py +324 -0
  119. rasa/core/brokers/pika.py +388 -0
  120. rasa/core/brokers/sql.py +86 -0
  121. rasa/core/channels/__init__.py +61 -0
  122. rasa/core/channels/botframework.py +338 -0
  123. rasa/core/channels/callback.py +84 -0
  124. rasa/core/channels/channel.py +456 -0
  125. rasa/core/channels/console.py +241 -0
  126. rasa/core/channels/development_inspector.py +197 -0
  127. rasa/core/channels/facebook.py +419 -0
  128. rasa/core/channels/hangouts.py +329 -0
  129. rasa/core/channels/inspector/.eslintrc.cjs +25 -0
  130. rasa/core/channels/inspector/.gitignore +23 -0
  131. rasa/core/channels/inspector/README.md +54 -0
  132. rasa/core/channels/inspector/assets/favicon.ico +0 -0
  133. rasa/core/channels/inspector/assets/rasa-chat.js +2 -0
  134. rasa/core/channels/inspector/custom.d.ts +3 -0
  135. rasa/core/channels/inspector/dist/assets/arc-632a63ec.js +1 -0
  136. rasa/core/channels/inspector/dist/assets/array-9f3ba611.js +1 -0
  137. rasa/core/channels/inspector/dist/assets/c4Diagram-d0fbc5ce-081e0df4.js +10 -0
  138. rasa/core/channels/inspector/dist/assets/classDiagram-936ed81e-3df0afc2.js +2 -0
  139. rasa/core/channels/inspector/dist/assets/classDiagram-v2-c3cb15f1-8c5ed31e.js +2 -0
  140. rasa/core/channels/inspector/dist/assets/createText-62fc7601-89c73b31.js +7 -0
  141. rasa/core/channels/inspector/dist/assets/edges-f2ad444c-4fc48c3e.js +4 -0
  142. rasa/core/channels/inspector/dist/assets/erDiagram-9d236eb7-907e0440.js +51 -0
  143. rasa/core/channels/inspector/dist/assets/flowDb-1972c806-9ec53a3c.js +6 -0
  144. rasa/core/channels/inspector/dist/assets/flowDiagram-7ea5b25a-41da787a.js +4 -0
  145. rasa/core/channels/inspector/dist/assets/flowDiagram-v2-855bc5b3-8bea338b.js +1 -0
  146. rasa/core/channels/inspector/dist/assets/flowchart-elk-definition-abe16c3d-ce370633.js +139 -0
  147. rasa/core/channels/inspector/dist/assets/ganttDiagram-9b5ea136-90a36523.js +266 -0
  148. rasa/core/channels/inspector/dist/assets/gitGraphDiagram-99d0ae7c-41e1aa3f.js +70 -0
  149. rasa/core/channels/inspector/dist/assets/ibm-plex-mono-v4-latin-regular-128cfa44.ttf +0 -0
  150. rasa/core/channels/inspector/dist/assets/ibm-plex-mono-v4-latin-regular-21dbcb97.woff +0 -0
  151. rasa/core/channels/inspector/dist/assets/ibm-plex-mono-v4-latin-regular-222b5e26.svg +329 -0
  152. rasa/core/channels/inspector/dist/assets/ibm-plex-mono-v4-latin-regular-9ad89b2a.woff2 +0 -0
  153. rasa/core/channels/inspector/dist/assets/index-2c4b9a3b-e6f2af62.js +1 -0
  154. rasa/core/channels/inspector/dist/assets/index-3ee28881.css +1 -0
  155. rasa/core/channels/inspector/dist/assets/index-e793d777.js +1317 -0
  156. rasa/core/channels/inspector/dist/assets/infoDiagram-736b4530-8ceba4db.js +7 -0
  157. rasa/core/channels/inspector/dist/assets/init-77b53fdd.js +1 -0
  158. rasa/core/channels/inspector/dist/assets/journeyDiagram-df861f2b-960d3809.js +139 -0
  159. rasa/core/channels/inspector/dist/assets/lato-v14-latin-700-60c05ee4.woff +0 -0
  160. rasa/core/channels/inspector/dist/assets/lato-v14-latin-700-8335d9b8.svg +438 -0
  161. rasa/core/channels/inspector/dist/assets/lato-v14-latin-700-9cc39c75.ttf +0 -0
  162. rasa/core/channels/inspector/dist/assets/lato-v14-latin-700-ead13ccf.woff2 +0 -0
  163. rasa/core/channels/inspector/dist/assets/lato-v14-latin-regular-16705655.woff2 +0 -0
  164. rasa/core/channels/inspector/dist/assets/lato-v14-latin-regular-5aeb07f9.woff +0 -0
  165. rasa/core/channels/inspector/dist/assets/lato-v14-latin-regular-9c459044.ttf +0 -0
  166. rasa/core/channels/inspector/dist/assets/lato-v14-latin-regular-9e2898a4.svg +435 -0
  167. rasa/core/channels/inspector/dist/assets/layout-498807d8.js +1 -0
  168. rasa/core/channels/inspector/dist/assets/line-eeccc4e2.js +1 -0
  169. rasa/core/channels/inspector/dist/assets/linear-8a078617.js +1 -0
  170. rasa/core/channels/inspector/dist/assets/mindmap-definition-beec6740-396d17dd.js +109 -0
  171. rasa/core/channels/inspector/dist/assets/ordinal-ba9b4969.js +1 -0
  172. rasa/core/channels/inspector/dist/assets/path-53f90ab3.js +1 -0
  173. rasa/core/channels/inspector/dist/assets/pieDiagram-dbbf0591-dc9b5e1b.js +35 -0
  174. rasa/core/channels/inspector/dist/assets/quadrantDiagram-4d7f4fd6-a08cba6d.js +7 -0
  175. rasa/core/channels/inspector/dist/assets/requirementDiagram-6fc4c22a-87242b9e.js +52 -0
  176. rasa/core/channels/inspector/dist/assets/sankeyDiagram-8f13d901-53f6f391.js +8 -0
  177. rasa/core/channels/inspector/dist/assets/sequenceDiagram-b655622a-715c9c20.js +122 -0
  178. rasa/core/channels/inspector/dist/assets/stateDiagram-59f0c015-2e8fb31f.js +1 -0
  179. rasa/core/channels/inspector/dist/assets/stateDiagram-v2-2b26beab-7e2d2aa0.js +1 -0
  180. rasa/core/channels/inspector/dist/assets/styles-080da4f6-4420cea6.js +110 -0
  181. rasa/core/channels/inspector/dist/assets/styles-3dcbcfbf-28676cf4.js +159 -0
  182. rasa/core/channels/inspector/dist/assets/styles-9c745c82-cef936a6.js +207 -0
  183. rasa/core/channels/inspector/dist/assets/svgDrawCommon-4835440b-151251e9.js +1 -0
  184. rasa/core/channels/inspector/dist/assets/timeline-definition-5b62e21b-0d39bdb2.js +61 -0
  185. rasa/core/channels/inspector/dist/assets/xychartDiagram-2b33534f-a03fa445.js +7 -0
  186. rasa/core/channels/inspector/dist/index.html +44 -0
  187. rasa/core/channels/inspector/index.html +42 -0
  188. rasa/core/channels/inspector/jest.config.ts +13 -0
  189. rasa/core/channels/inspector/package.json +52 -0
  190. rasa/core/channels/inspector/setupTests.ts +2 -0
  191. rasa/core/channels/inspector/src/App.tsx +217 -0
  192. rasa/core/channels/inspector/src/components/Chat.tsx +95 -0
  193. rasa/core/channels/inspector/src/components/DiagramFlow.tsx +108 -0
  194. rasa/core/channels/inspector/src/components/DialogueInformation.tsx +187 -0
  195. rasa/core/channels/inspector/src/components/DialogueStack.tsx +136 -0
  196. rasa/core/channels/inspector/src/components/ExpandIcon.tsx +16 -0
  197. rasa/core/channels/inspector/src/components/FullscreenButton.tsx +45 -0
  198. rasa/core/channels/inspector/src/components/LoadingSpinner.tsx +22 -0
  199. rasa/core/channels/inspector/src/components/NoActiveFlow.tsx +21 -0
  200. rasa/core/channels/inspector/src/components/RasaLogo.tsx +32 -0
  201. rasa/core/channels/inspector/src/components/SaraDiagrams.tsx +39 -0
  202. rasa/core/channels/inspector/src/components/Slots.tsx +91 -0
  203. rasa/core/channels/inspector/src/components/Welcome.tsx +54 -0
  204. rasa/core/channels/inspector/src/helpers/audiostream.ts +191 -0
  205. rasa/core/channels/inspector/src/helpers/formatters.test.ts +392 -0
  206. rasa/core/channels/inspector/src/helpers/formatters.ts +306 -0
  207. rasa/core/channels/inspector/src/helpers/utils.ts +127 -0
  208. rasa/core/channels/inspector/src/main.tsx +13 -0
  209. rasa/core/channels/inspector/src/theme/Button/Button.ts +29 -0
  210. rasa/core/channels/inspector/src/theme/Heading/Heading.ts +31 -0
  211. rasa/core/channels/inspector/src/theme/Input/Input.ts +27 -0
  212. rasa/core/channels/inspector/src/theme/Link/Link.ts +10 -0
  213. rasa/core/channels/inspector/src/theme/Modal/Modal.ts +47 -0
  214. rasa/core/channels/inspector/src/theme/Table/Table.tsx +38 -0
  215. rasa/core/channels/inspector/src/theme/Tooltip/Tooltip.ts +12 -0
  216. rasa/core/channels/inspector/src/theme/base/breakpoints.ts +8 -0
  217. rasa/core/channels/inspector/src/theme/base/colors.ts +88 -0
  218. rasa/core/channels/inspector/src/theme/base/fonts/fontFaces.css +29 -0
  219. rasa/core/channels/inspector/src/theme/base/fonts/ibm-plex-mono-v4-latin/ibm-plex-mono-v4-latin-regular.eot +0 -0
  220. rasa/core/channels/inspector/src/theme/base/fonts/ibm-plex-mono-v4-latin/ibm-plex-mono-v4-latin-regular.svg +329 -0
  221. rasa/core/channels/inspector/src/theme/base/fonts/ibm-plex-mono-v4-latin/ibm-plex-mono-v4-latin-regular.ttf +0 -0
  222. rasa/core/channels/inspector/src/theme/base/fonts/ibm-plex-mono-v4-latin/ibm-plex-mono-v4-latin-regular.woff +0 -0
  223. rasa/core/channels/inspector/src/theme/base/fonts/ibm-plex-mono-v4-latin/ibm-plex-mono-v4-latin-regular.woff2 +0 -0
  224. rasa/core/channels/inspector/src/theme/base/fonts/lato-v14-latin/lato-v14-latin-700.eot +0 -0
  225. rasa/core/channels/inspector/src/theme/base/fonts/lato-v14-latin/lato-v14-latin-700.svg +438 -0
  226. rasa/core/channels/inspector/src/theme/base/fonts/lato-v14-latin/lato-v14-latin-700.ttf +0 -0
  227. rasa/core/channels/inspector/src/theme/base/fonts/lato-v14-latin/lato-v14-latin-700.woff +0 -0
  228. rasa/core/channels/inspector/src/theme/base/fonts/lato-v14-latin/lato-v14-latin-700.woff2 +0 -0
  229. rasa/core/channels/inspector/src/theme/base/fonts/lato-v14-latin/lato-v14-latin-regular.eot +0 -0
  230. rasa/core/channels/inspector/src/theme/base/fonts/lato-v14-latin/lato-v14-latin-regular.svg +435 -0
  231. rasa/core/channels/inspector/src/theme/base/fonts/lato-v14-latin/lato-v14-latin-regular.ttf +0 -0
  232. rasa/core/channels/inspector/src/theme/base/fonts/lato-v14-latin/lato-v14-latin-regular.woff +0 -0
  233. rasa/core/channels/inspector/src/theme/base/fonts/lato-v14-latin/lato-v14-latin-regular.woff2 +0 -0
  234. rasa/core/channels/inspector/src/theme/base/radii.ts +9 -0
  235. rasa/core/channels/inspector/src/theme/base/shadows.ts +7 -0
  236. rasa/core/channels/inspector/src/theme/base/sizes.ts +7 -0
  237. rasa/core/channels/inspector/src/theme/base/space.ts +15 -0
  238. rasa/core/channels/inspector/src/theme/base/styles.ts +13 -0
  239. rasa/core/channels/inspector/src/theme/base/typography.ts +24 -0
  240. rasa/core/channels/inspector/src/theme/base/zIndices.ts +19 -0
  241. rasa/core/channels/inspector/src/theme/index.ts +101 -0
  242. rasa/core/channels/inspector/src/types.ts +84 -0
  243. rasa/core/channels/inspector/src/vite-env.d.ts +1 -0
  244. rasa/core/channels/inspector/tests/__mocks__/fileMock.ts +1 -0
  245. rasa/core/channels/inspector/tests/__mocks__/matchMedia.ts +16 -0
  246. rasa/core/channels/inspector/tests/__mocks__/styleMock.ts +1 -0
  247. rasa/core/channels/inspector/tests/renderWithProviders.tsx +14 -0
  248. rasa/core/channels/inspector/tsconfig.json +26 -0
  249. rasa/core/channels/inspector/tsconfig.node.json +10 -0
  250. rasa/core/channels/inspector/vite.config.ts +8 -0
  251. rasa/core/channels/inspector/yarn.lock +6249 -0
  252. rasa/core/channels/mattermost.py +229 -0
  253. rasa/core/channels/rasa_chat.py +126 -0
  254. rasa/core/channels/rest.py +230 -0
  255. rasa/core/channels/rocketchat.py +174 -0
  256. rasa/core/channels/slack.py +620 -0
  257. rasa/core/channels/socketio.py +301 -0
  258. rasa/core/channels/telegram.py +298 -0
  259. rasa/core/channels/twilio.py +169 -0
  260. rasa/core/channels/vier_cvg.py +374 -0
  261. rasa/core/channels/voice_ready/__init__.py +0 -0
  262. rasa/core/channels/voice_ready/audiocodes.py +501 -0
  263. rasa/core/channels/voice_ready/jambonz.py +121 -0
  264. rasa/core/channels/voice_ready/jambonz_protocol.py +396 -0
  265. rasa/core/channels/voice_ready/twilio_voice.py +403 -0
  266. rasa/core/channels/voice_ready/utils.py +37 -0
  267. rasa/core/channels/voice_stream/__init__.py +0 -0
  268. rasa/core/channels/voice_stream/asr/__init__.py +0 -0
  269. rasa/core/channels/voice_stream/asr/asr_engine.py +89 -0
  270. rasa/core/channels/voice_stream/asr/asr_event.py +18 -0
  271. rasa/core/channels/voice_stream/asr/azure.py +130 -0
  272. rasa/core/channels/voice_stream/asr/deepgram.py +90 -0
  273. rasa/core/channels/voice_stream/audio_bytes.py +8 -0
  274. rasa/core/channels/voice_stream/browser_audio.py +107 -0
  275. rasa/core/channels/voice_stream/call_state.py +23 -0
  276. rasa/core/channels/voice_stream/tts/__init__.py +0 -0
  277. rasa/core/channels/voice_stream/tts/azure.py +106 -0
  278. rasa/core/channels/voice_stream/tts/cartesia.py +118 -0
  279. rasa/core/channels/voice_stream/tts/tts_cache.py +27 -0
  280. rasa/core/channels/voice_stream/tts/tts_engine.py +58 -0
  281. rasa/core/channels/voice_stream/twilio_media_streams.py +173 -0
  282. rasa/core/channels/voice_stream/util.py +57 -0
  283. rasa/core/channels/voice_stream/voice_channel.py +427 -0
  284. rasa/core/channels/webexteams.py +134 -0
  285. rasa/core/concurrent_lock_store.py +210 -0
  286. rasa/core/constants.py +112 -0
  287. rasa/core/evaluation/__init__.py +0 -0
  288. rasa/core/evaluation/marker.py +267 -0
  289. rasa/core/evaluation/marker_base.py +923 -0
  290. rasa/core/evaluation/marker_stats.py +293 -0
  291. rasa/core/evaluation/marker_tracker_loader.py +103 -0
  292. rasa/core/exceptions.py +29 -0
  293. rasa/core/exporter.py +284 -0
  294. rasa/core/featurizers/__init__.py +0 -0
  295. rasa/core/featurizers/precomputation.py +410 -0
  296. rasa/core/featurizers/single_state_featurizer.py +421 -0
  297. rasa/core/featurizers/tracker_featurizers.py +1262 -0
  298. rasa/core/http_interpreter.py +89 -0
  299. rasa/core/information_retrieval/__init__.py +7 -0
  300. rasa/core/information_retrieval/faiss.py +124 -0
  301. rasa/core/information_retrieval/information_retrieval.py +137 -0
  302. rasa/core/information_retrieval/milvus.py +59 -0
  303. rasa/core/information_retrieval/qdrant.py +96 -0
  304. rasa/core/jobs.py +63 -0
  305. rasa/core/lock.py +139 -0
  306. rasa/core/lock_store.py +343 -0
  307. rasa/core/migrate.py +403 -0
  308. rasa/core/nlg/__init__.py +3 -0
  309. rasa/core/nlg/callback.py +146 -0
  310. rasa/core/nlg/contextual_response_rephraser.py +320 -0
  311. rasa/core/nlg/generator.py +230 -0
  312. rasa/core/nlg/interpolator.py +143 -0
  313. rasa/core/nlg/response.py +155 -0
  314. rasa/core/nlg/summarize.py +70 -0
  315. rasa/core/persistor.py +538 -0
  316. rasa/core/policies/__init__.py +0 -0
  317. rasa/core/policies/ensemble.py +329 -0
  318. rasa/core/policies/enterprise_search_policy.py +905 -0
  319. rasa/core/policies/enterprise_search_prompt_template.jinja2 +25 -0
  320. rasa/core/policies/enterprise_search_prompt_with_citation_template.jinja2 +60 -0
  321. rasa/core/policies/flow_policy.py +205 -0
  322. rasa/core/policies/flows/__init__.py +0 -0
  323. rasa/core/policies/flows/flow_exceptions.py +44 -0
  324. rasa/core/policies/flows/flow_executor.py +754 -0
  325. rasa/core/policies/flows/flow_step_result.py +43 -0
  326. rasa/core/policies/intentless_policy.py +1031 -0
  327. rasa/core/policies/intentless_prompt_template.jinja2 +22 -0
  328. rasa/core/policies/memoization.py +538 -0
  329. rasa/core/policies/policy.py +725 -0
  330. rasa/core/policies/rule_policy.py +1273 -0
  331. rasa/core/policies/ted_policy.py +2169 -0
  332. rasa/core/policies/unexpected_intent_policy.py +1022 -0
  333. rasa/core/processor.py +1465 -0
  334. rasa/core/run.py +342 -0
  335. rasa/core/secrets_manager/__init__.py +0 -0
  336. rasa/core/secrets_manager/constants.py +36 -0
  337. rasa/core/secrets_manager/endpoints.py +391 -0
  338. rasa/core/secrets_manager/factory.py +241 -0
  339. rasa/core/secrets_manager/secret_manager.py +262 -0
  340. rasa/core/secrets_manager/vault.py +584 -0
  341. rasa/core/test.py +1335 -0
  342. rasa/core/tracker_store.py +1703 -0
  343. rasa/core/train.py +105 -0
  344. rasa/core/training/__init__.py +89 -0
  345. rasa/core/training/converters/__init__.py +0 -0
  346. rasa/core/training/converters/responses_prefix_converter.py +119 -0
  347. rasa/core/training/interactive.py +1744 -0
  348. rasa/core/training/story_conflict.py +381 -0
  349. rasa/core/training/training.py +93 -0
  350. rasa/core/utils.py +366 -0
  351. rasa/core/visualize.py +70 -0
  352. rasa/dialogue_understanding/__init__.py +0 -0
  353. rasa/dialogue_understanding/coexistence/__init__.py +0 -0
  354. rasa/dialogue_understanding/coexistence/constants.py +4 -0
  355. rasa/dialogue_understanding/coexistence/intent_based_router.py +196 -0
  356. rasa/dialogue_understanding/coexistence/llm_based_router.py +327 -0
  357. rasa/dialogue_understanding/coexistence/router_template.jinja2 +12 -0
  358. rasa/dialogue_understanding/commands/__init__.py +61 -0
  359. rasa/dialogue_understanding/commands/can_not_handle_command.py +70 -0
  360. rasa/dialogue_understanding/commands/cancel_flow_command.py +125 -0
  361. rasa/dialogue_understanding/commands/change_flow_command.py +44 -0
  362. rasa/dialogue_understanding/commands/chit_chat_answer_command.py +57 -0
  363. rasa/dialogue_understanding/commands/clarify_command.py +86 -0
  364. rasa/dialogue_understanding/commands/command.py +85 -0
  365. rasa/dialogue_understanding/commands/correct_slots_command.py +297 -0
  366. rasa/dialogue_understanding/commands/error_command.py +79 -0
  367. rasa/dialogue_understanding/commands/free_form_answer_command.py +9 -0
  368. rasa/dialogue_understanding/commands/handle_code_change_command.py +73 -0
  369. rasa/dialogue_understanding/commands/human_handoff_command.py +66 -0
  370. rasa/dialogue_understanding/commands/knowledge_answer_command.py +57 -0
  371. rasa/dialogue_understanding/commands/noop_command.py +54 -0
  372. rasa/dialogue_understanding/commands/repeat_bot_messages_command.py +60 -0
  373. rasa/dialogue_understanding/commands/restart_command.py +58 -0
  374. rasa/dialogue_understanding/commands/session_end_command.py +61 -0
  375. rasa/dialogue_understanding/commands/session_start_command.py +59 -0
  376. rasa/dialogue_understanding/commands/set_slot_command.py +160 -0
  377. rasa/dialogue_understanding/commands/skip_question_command.py +75 -0
  378. rasa/dialogue_understanding/commands/start_flow_command.py +107 -0
  379. rasa/dialogue_understanding/commands/user_silence_command.py +59 -0
  380. rasa/dialogue_understanding/commands/utils.py +45 -0
  381. rasa/dialogue_understanding/generator/__init__.py +21 -0
  382. rasa/dialogue_understanding/generator/command_generator.py +343 -0
  383. rasa/dialogue_understanding/generator/constants.py +27 -0
  384. rasa/dialogue_understanding/generator/flow_document_template.jinja2 +4 -0
  385. rasa/dialogue_understanding/generator/flow_retrieval.py +466 -0
  386. rasa/dialogue_understanding/generator/llm_based_command_generator.py +500 -0
  387. rasa/dialogue_understanding/generator/llm_command_generator.py +67 -0
  388. rasa/dialogue_understanding/generator/multi_step/__init__.py +0 -0
  389. rasa/dialogue_understanding/generator/multi_step/fill_slots_prompt.jinja2 +62 -0
  390. rasa/dialogue_understanding/generator/multi_step/handle_flows_prompt.jinja2 +38 -0
  391. rasa/dialogue_understanding/generator/multi_step/multi_step_llm_command_generator.py +893 -0
  392. rasa/dialogue_understanding/generator/nlu_command_adapter.py +258 -0
  393. rasa/dialogue_understanding/generator/single_step/__init__.py +0 -0
  394. rasa/dialogue_understanding/generator/single_step/command_prompt_template.jinja2 +60 -0
  395. rasa/dialogue_understanding/generator/single_step/single_step_llm_command_generator.py +478 -0
  396. rasa/dialogue_understanding/patterns/__init__.py +0 -0
  397. rasa/dialogue_understanding/patterns/cancel.py +111 -0
  398. rasa/dialogue_understanding/patterns/cannot_handle.py +43 -0
  399. rasa/dialogue_understanding/patterns/chitchat.py +37 -0
  400. rasa/dialogue_understanding/patterns/clarify.py +97 -0
  401. rasa/dialogue_understanding/patterns/code_change.py +41 -0
  402. rasa/dialogue_understanding/patterns/collect_information.py +90 -0
  403. rasa/dialogue_understanding/patterns/completed.py +40 -0
  404. rasa/dialogue_understanding/patterns/continue_interrupted.py +42 -0
  405. rasa/dialogue_understanding/patterns/correction.py +278 -0
  406. rasa/dialogue_understanding/patterns/default_flows_for_patterns.yml +301 -0
  407. rasa/dialogue_understanding/patterns/human_handoff.py +37 -0
  408. rasa/dialogue_understanding/patterns/internal_error.py +47 -0
  409. rasa/dialogue_understanding/patterns/repeat.py +37 -0
  410. rasa/dialogue_understanding/patterns/restart.py +37 -0
  411. rasa/dialogue_understanding/patterns/search.py +37 -0
  412. rasa/dialogue_understanding/patterns/session_start.py +37 -0
  413. rasa/dialogue_understanding/patterns/skip_question.py +38 -0
  414. rasa/dialogue_understanding/patterns/user_silence.py +37 -0
  415. rasa/dialogue_understanding/processor/__init__.py +0 -0
  416. rasa/dialogue_understanding/processor/command_processor.py +720 -0
  417. rasa/dialogue_understanding/processor/command_processor_component.py +43 -0
  418. rasa/dialogue_understanding/stack/__init__.py +0 -0
  419. rasa/dialogue_understanding/stack/dialogue_stack.py +178 -0
  420. rasa/dialogue_understanding/stack/frames/__init__.py +19 -0
  421. rasa/dialogue_understanding/stack/frames/chit_chat_frame.py +27 -0
  422. rasa/dialogue_understanding/stack/frames/dialogue_stack_frame.py +137 -0
  423. rasa/dialogue_understanding/stack/frames/flow_stack_frame.py +157 -0
  424. rasa/dialogue_understanding/stack/frames/pattern_frame.py +10 -0
  425. rasa/dialogue_understanding/stack/frames/search_frame.py +27 -0
  426. rasa/dialogue_understanding/stack/utils.py +211 -0
  427. rasa/e2e_test/__init__.py +0 -0
  428. rasa/e2e_test/aggregate_test_stats_calculator.py +134 -0
  429. rasa/e2e_test/assertions.py +1345 -0
  430. rasa/e2e_test/assertions_schema.yml +129 -0
  431. rasa/e2e_test/constants.py +31 -0
  432. rasa/e2e_test/e2e_config.py +220 -0
  433. rasa/e2e_test/e2e_config_schema.yml +26 -0
  434. rasa/e2e_test/e2e_test_case.py +568 -0
  435. rasa/e2e_test/e2e_test_converter.py +363 -0
  436. rasa/e2e_test/e2e_test_converter_prompt.jinja2 +70 -0
  437. rasa/e2e_test/e2e_test_coverage_report.py +364 -0
  438. rasa/e2e_test/e2e_test_result.py +54 -0
  439. rasa/e2e_test/e2e_test_runner.py +1190 -0
  440. rasa/e2e_test/e2e_test_schema.yml +181 -0
  441. rasa/e2e_test/pykwalify_extensions.py +39 -0
  442. rasa/e2e_test/stub_custom_action.py +70 -0
  443. rasa/e2e_test/utils/__init__.py +0 -0
  444. rasa/e2e_test/utils/e2e_yaml_utils.py +55 -0
  445. rasa/e2e_test/utils/io.py +598 -0
  446. rasa/e2e_test/utils/validation.py +80 -0
  447. rasa/engine/__init__.py +0 -0
  448. rasa/engine/caching.py +463 -0
  449. rasa/engine/constants.py +17 -0
  450. rasa/engine/exceptions.py +14 -0
  451. rasa/engine/graph.py +642 -0
  452. rasa/engine/loader.py +48 -0
  453. rasa/engine/recipes/__init__.py +0 -0
  454. rasa/engine/recipes/config_files/default_config.yml +41 -0
  455. rasa/engine/recipes/default_components.py +97 -0
  456. rasa/engine/recipes/default_recipe.py +1258 -0
  457. rasa/engine/recipes/graph_recipe.py +78 -0
  458. rasa/engine/recipes/recipe.py +93 -0
  459. rasa/engine/runner/__init__.py +0 -0
  460. rasa/engine/runner/dask.py +250 -0
  461. rasa/engine/runner/interface.py +49 -0
  462. rasa/engine/storage/__init__.py +0 -0
  463. rasa/engine/storage/local_model_storage.py +244 -0
  464. rasa/engine/storage/resource.py +110 -0
  465. rasa/engine/storage/storage.py +199 -0
  466. rasa/engine/training/__init__.py +0 -0
  467. rasa/engine/training/components.py +176 -0
  468. rasa/engine/training/fingerprinting.py +64 -0
  469. rasa/engine/training/graph_trainer.py +256 -0
  470. rasa/engine/training/hooks.py +164 -0
  471. rasa/engine/validation.py +1451 -0
  472. rasa/env.py +14 -0
  473. rasa/exceptions.py +69 -0
  474. rasa/graph_components/__init__.py +0 -0
  475. rasa/graph_components/converters/__init__.py +0 -0
  476. rasa/graph_components/converters/nlu_message_converter.py +48 -0
  477. rasa/graph_components/providers/__init__.py +0 -0
  478. rasa/graph_components/providers/domain_for_core_training_provider.py +87 -0
  479. rasa/graph_components/providers/domain_provider.py +71 -0
  480. rasa/graph_components/providers/flows_provider.py +74 -0
  481. rasa/graph_components/providers/forms_provider.py +44 -0
  482. rasa/graph_components/providers/nlu_training_data_provider.py +56 -0
  483. rasa/graph_components/providers/responses_provider.py +44 -0
  484. rasa/graph_components/providers/rule_only_provider.py +49 -0
  485. rasa/graph_components/providers/story_graph_provider.py +96 -0
  486. rasa/graph_components/providers/training_tracker_provider.py +55 -0
  487. rasa/graph_components/validators/__init__.py +0 -0
  488. rasa/graph_components/validators/default_recipe_validator.py +550 -0
  489. rasa/graph_components/validators/finetuning_validator.py +302 -0
  490. rasa/hooks.py +112 -0
  491. rasa/jupyter.py +63 -0
  492. rasa/llm_fine_tuning/__init__.py +0 -0
  493. rasa/llm_fine_tuning/annotation_module.py +241 -0
  494. rasa/llm_fine_tuning/conversations.py +144 -0
  495. rasa/llm_fine_tuning/llm_data_preparation_module.py +178 -0
  496. rasa/llm_fine_tuning/paraphrasing/__init__.py +0 -0
  497. rasa/llm_fine_tuning/paraphrasing/conversation_rephraser.py +281 -0
  498. rasa/llm_fine_tuning/paraphrasing/default_rephrase_prompt_template.jina2 +44 -0
  499. rasa/llm_fine_tuning/paraphrasing/rephrase_validator.py +121 -0
  500. rasa/llm_fine_tuning/paraphrasing/rephrased_user_message.py +10 -0
  501. rasa/llm_fine_tuning/paraphrasing_module.py +128 -0
  502. rasa/llm_fine_tuning/storage.py +174 -0
  503. rasa/llm_fine_tuning/train_test_split_module.py +441 -0
  504. rasa/markers/__init__.py +0 -0
  505. rasa/markers/marker.py +269 -0
  506. rasa/markers/marker_base.py +828 -0
  507. rasa/markers/upload.py +74 -0
  508. rasa/markers/validate.py +21 -0
  509. rasa/model.py +118 -0
  510. rasa/model_manager/__init__.py +0 -0
  511. rasa/model_manager/config.py +40 -0
  512. rasa/model_manager/model_api.py +559 -0
  513. rasa/model_manager/runner_service.py +286 -0
  514. rasa/model_manager/socket_bridge.py +146 -0
  515. rasa/model_manager/studio_jwt_auth.py +86 -0
  516. rasa/model_manager/trainer_service.py +325 -0
  517. rasa/model_manager/utils.py +87 -0
  518. rasa/model_manager/warm_rasa_process.py +187 -0
  519. rasa/model_service.py +112 -0
  520. rasa/model_testing.py +457 -0
  521. rasa/model_training.py +595 -0
  522. rasa/nlu/__init__.py +7 -0
  523. rasa/nlu/classifiers/__init__.py +3 -0
  524. rasa/nlu/classifiers/classifier.py +5 -0
  525. rasa/nlu/classifiers/diet_classifier.py +1881 -0
  526. rasa/nlu/classifiers/fallback_classifier.py +192 -0
  527. rasa/nlu/classifiers/keyword_intent_classifier.py +188 -0
  528. rasa/nlu/classifiers/logistic_regression_classifier.py +253 -0
  529. rasa/nlu/classifiers/mitie_intent_classifier.py +156 -0
  530. rasa/nlu/classifiers/regex_message_handler.py +56 -0
  531. rasa/nlu/classifiers/sklearn_intent_classifier.py +330 -0
  532. rasa/nlu/constants.py +77 -0
  533. rasa/nlu/convert.py +40 -0
  534. rasa/nlu/emulators/__init__.py +0 -0
  535. rasa/nlu/emulators/dialogflow.py +55 -0
  536. rasa/nlu/emulators/emulator.py +49 -0
  537. rasa/nlu/emulators/luis.py +86 -0
  538. rasa/nlu/emulators/no_emulator.py +10 -0
  539. rasa/nlu/emulators/wit.py +56 -0
  540. rasa/nlu/extractors/__init__.py +0 -0
  541. rasa/nlu/extractors/crf_entity_extractor.py +715 -0
  542. rasa/nlu/extractors/duckling_entity_extractor.py +206 -0
  543. rasa/nlu/extractors/entity_synonyms.py +178 -0
  544. rasa/nlu/extractors/extractor.py +470 -0
  545. rasa/nlu/extractors/mitie_entity_extractor.py +293 -0
  546. rasa/nlu/extractors/regex_entity_extractor.py +220 -0
  547. rasa/nlu/extractors/spacy_entity_extractor.py +95 -0
  548. rasa/nlu/featurizers/__init__.py +0 -0
  549. rasa/nlu/featurizers/dense_featurizer/__init__.py +0 -0
  550. rasa/nlu/featurizers/dense_featurizer/convert_featurizer.py +445 -0
  551. rasa/nlu/featurizers/dense_featurizer/dense_featurizer.py +57 -0
  552. rasa/nlu/featurizers/dense_featurizer/lm_featurizer.py +768 -0
  553. rasa/nlu/featurizers/dense_featurizer/mitie_featurizer.py +170 -0
  554. rasa/nlu/featurizers/dense_featurizer/spacy_featurizer.py +132 -0
  555. rasa/nlu/featurizers/featurizer.py +89 -0
  556. rasa/nlu/featurizers/sparse_featurizer/__init__.py +0 -0
  557. rasa/nlu/featurizers/sparse_featurizer/count_vectors_featurizer.py +867 -0
  558. rasa/nlu/featurizers/sparse_featurizer/lexical_syntactic_featurizer.py +571 -0
  559. rasa/nlu/featurizers/sparse_featurizer/regex_featurizer.py +271 -0
  560. rasa/nlu/featurizers/sparse_featurizer/sparse_featurizer.py +9 -0
  561. rasa/nlu/model.py +24 -0
  562. rasa/nlu/run.py +27 -0
  563. rasa/nlu/selectors/__init__.py +0 -0
  564. rasa/nlu/selectors/response_selector.py +987 -0
  565. rasa/nlu/test.py +1940 -0
  566. rasa/nlu/tokenizers/__init__.py +0 -0
  567. rasa/nlu/tokenizers/jieba_tokenizer.py +148 -0
  568. rasa/nlu/tokenizers/mitie_tokenizer.py +75 -0
  569. rasa/nlu/tokenizers/spacy_tokenizer.py +72 -0
  570. rasa/nlu/tokenizers/tokenizer.py +239 -0
  571. rasa/nlu/tokenizers/whitespace_tokenizer.py +95 -0
  572. rasa/nlu/utils/__init__.py +35 -0
  573. rasa/nlu/utils/bilou_utils.py +462 -0
  574. rasa/nlu/utils/hugging_face/__init__.py +0 -0
  575. rasa/nlu/utils/hugging_face/registry.py +108 -0
  576. rasa/nlu/utils/hugging_face/transformers_pre_post_processors.py +311 -0
  577. rasa/nlu/utils/mitie_utils.py +113 -0
  578. rasa/nlu/utils/pattern_utils.py +168 -0
  579. rasa/nlu/utils/spacy_utils.py +310 -0
  580. rasa/plugin.py +90 -0
  581. rasa/server.py +1624 -0
  582. rasa/shared/__init__.py +0 -0
  583. rasa/shared/constants.py +310 -0
  584. rasa/shared/core/__init__.py +0 -0
  585. rasa/shared/core/command_payload_reader.py +109 -0
  586. rasa/shared/core/constants.py +180 -0
  587. rasa/shared/core/conversation.py +46 -0
  588. rasa/shared/core/domain.py +2172 -0
  589. rasa/shared/core/events.py +2559 -0
  590. rasa/shared/core/flows/__init__.py +7 -0
  591. rasa/shared/core/flows/flow.py +562 -0
  592. rasa/shared/core/flows/flow_path.py +84 -0
  593. rasa/shared/core/flows/flow_step.py +146 -0
  594. rasa/shared/core/flows/flow_step_links.py +319 -0
  595. rasa/shared/core/flows/flow_step_sequence.py +70 -0
  596. rasa/shared/core/flows/flows_list.py +258 -0
  597. rasa/shared/core/flows/flows_yaml_schema.json +303 -0
  598. rasa/shared/core/flows/nlu_trigger.py +117 -0
  599. rasa/shared/core/flows/steps/__init__.py +24 -0
  600. rasa/shared/core/flows/steps/action.py +56 -0
  601. rasa/shared/core/flows/steps/call.py +64 -0
  602. rasa/shared/core/flows/steps/collect.py +112 -0
  603. rasa/shared/core/flows/steps/constants.py +5 -0
  604. rasa/shared/core/flows/steps/continuation.py +36 -0
  605. rasa/shared/core/flows/steps/end.py +22 -0
  606. rasa/shared/core/flows/steps/internal.py +44 -0
  607. rasa/shared/core/flows/steps/link.py +51 -0
  608. rasa/shared/core/flows/steps/no_operation.py +48 -0
  609. rasa/shared/core/flows/steps/set_slots.py +50 -0
  610. rasa/shared/core/flows/steps/start.py +30 -0
  611. rasa/shared/core/flows/utils.py +39 -0
  612. rasa/shared/core/flows/validation.py +735 -0
  613. rasa/shared/core/flows/yaml_flows_io.py +405 -0
  614. rasa/shared/core/generator.py +908 -0
  615. rasa/shared/core/slot_mappings.py +526 -0
  616. rasa/shared/core/slots.py +654 -0
  617. rasa/shared/core/trackers.py +1183 -0
  618. rasa/shared/core/training_data/__init__.py +0 -0
  619. rasa/shared/core/training_data/loading.py +89 -0
  620. rasa/shared/core/training_data/story_reader/__init__.py +0 -0
  621. rasa/shared/core/training_data/story_reader/story_reader.py +129 -0
  622. rasa/shared/core/training_data/story_reader/story_step_builder.py +168 -0
  623. rasa/shared/core/training_data/story_reader/yaml_story_reader.py +888 -0
  624. rasa/shared/core/training_data/story_writer/__init__.py +0 -0
  625. rasa/shared/core/training_data/story_writer/story_writer.py +76 -0
  626. rasa/shared/core/training_data/story_writer/yaml_story_writer.py +444 -0
  627. rasa/shared/core/training_data/structures.py +858 -0
  628. rasa/shared/core/training_data/visualization.html +146 -0
  629. rasa/shared/core/training_data/visualization.py +603 -0
  630. rasa/shared/data.py +249 -0
  631. rasa/shared/engine/__init__.py +0 -0
  632. rasa/shared/engine/caching.py +26 -0
  633. rasa/shared/exceptions.py +167 -0
  634. rasa/shared/importers/__init__.py +0 -0
  635. rasa/shared/importers/importer.py +770 -0
  636. rasa/shared/importers/multi_project.py +215 -0
  637. rasa/shared/importers/rasa.py +108 -0
  638. rasa/shared/importers/remote_importer.py +196 -0
  639. rasa/shared/importers/utils.py +36 -0
  640. rasa/shared/nlu/__init__.py +0 -0
  641. rasa/shared/nlu/constants.py +49 -0
  642. rasa/shared/nlu/interpreter.py +10 -0
  643. rasa/shared/nlu/training_data/__init__.py +0 -0
  644. rasa/shared/nlu/training_data/entities_parser.py +208 -0
  645. rasa/shared/nlu/training_data/features.py +492 -0
  646. rasa/shared/nlu/training_data/formats/__init__.py +10 -0
  647. rasa/shared/nlu/training_data/formats/dialogflow.py +163 -0
  648. rasa/shared/nlu/training_data/formats/luis.py +87 -0
  649. rasa/shared/nlu/training_data/formats/rasa.py +135 -0
  650. rasa/shared/nlu/training_data/formats/rasa_yaml.py +618 -0
  651. rasa/shared/nlu/training_data/formats/readerwriter.py +244 -0
  652. rasa/shared/nlu/training_data/formats/wit.py +52 -0
  653. rasa/shared/nlu/training_data/loading.py +137 -0
  654. rasa/shared/nlu/training_data/lookup_tables_parser.py +30 -0
  655. rasa/shared/nlu/training_data/message.py +490 -0
  656. rasa/shared/nlu/training_data/schemas/__init__.py +0 -0
  657. rasa/shared/nlu/training_data/schemas/data_schema.py +85 -0
  658. rasa/shared/nlu/training_data/schemas/nlu.yml +53 -0
  659. rasa/shared/nlu/training_data/schemas/responses.yml +70 -0
  660. rasa/shared/nlu/training_data/synonyms_parser.py +42 -0
  661. rasa/shared/nlu/training_data/training_data.py +729 -0
  662. rasa/shared/nlu/training_data/util.py +223 -0
  663. rasa/shared/providers/__init__.py +0 -0
  664. rasa/shared/providers/_configs/__init__.py +0 -0
  665. rasa/shared/providers/_configs/azure_openai_client_config.py +183 -0
  666. rasa/shared/providers/_configs/client_config.py +57 -0
  667. rasa/shared/providers/_configs/default_litellm_client_config.py +130 -0
  668. rasa/shared/providers/_configs/huggingface_local_embedding_client_config.py +234 -0
  669. rasa/shared/providers/_configs/litellm_router_client_config.py +220 -0
  670. rasa/shared/providers/_configs/model_group_config.py +167 -0
  671. rasa/shared/providers/_configs/openai_client_config.py +175 -0
  672. rasa/shared/providers/_configs/rasa_llm_client_config.py +73 -0
  673. rasa/shared/providers/_configs/self_hosted_llm_client_config.py +177 -0
  674. rasa/shared/providers/_configs/utils.py +117 -0
  675. rasa/shared/providers/_ssl_verification_utils.py +124 -0
  676. rasa/shared/providers/_utils.py +79 -0
  677. rasa/shared/providers/embedding/__init__.py +0 -0
  678. rasa/shared/providers/embedding/_base_litellm_embedding_client.py +243 -0
  679. rasa/shared/providers/embedding/_langchain_embedding_client_adapter.py +74 -0
  680. rasa/shared/providers/embedding/azure_openai_embedding_client.py +310 -0
  681. rasa/shared/providers/embedding/default_litellm_embedding_client.py +126 -0
  682. rasa/shared/providers/embedding/embedding_client.py +90 -0
  683. rasa/shared/providers/embedding/embedding_response.py +41 -0
  684. rasa/shared/providers/embedding/huggingface_local_embedding_client.py +191 -0
  685. rasa/shared/providers/embedding/litellm_router_embedding_client.py +135 -0
  686. rasa/shared/providers/embedding/openai_embedding_client.py +172 -0
  687. rasa/shared/providers/llm/__init__.py +0 -0
  688. rasa/shared/providers/llm/_base_litellm_client.py +263 -0
  689. rasa/shared/providers/llm/azure_openai_llm_client.py +359 -0
  690. rasa/shared/providers/llm/default_litellm_llm_client.py +108 -0
  691. rasa/shared/providers/llm/litellm_router_llm_client.py +182 -0
  692. rasa/shared/providers/llm/llm_client.py +76 -0
  693. rasa/shared/providers/llm/llm_response.py +50 -0
  694. rasa/shared/providers/llm/openai_llm_client.py +155 -0
  695. rasa/shared/providers/llm/rasa_llm_client.py +112 -0
  696. rasa/shared/providers/llm/self_hosted_llm_client.py +269 -0
  697. rasa/shared/providers/mappings.py +94 -0
  698. rasa/shared/providers/router/__init__.py +0 -0
  699. rasa/shared/providers/router/_base_litellm_router_client.py +183 -0
  700. rasa/shared/providers/router/router_client.py +73 -0
  701. rasa/shared/utils/__init__.py +0 -0
  702. rasa/shared/utils/cli.py +102 -0
  703. rasa/shared/utils/common.py +324 -0
  704. rasa/shared/utils/constants.py +4 -0
  705. rasa/shared/utils/health_check/__init__.py +0 -0
  706. rasa/shared/utils/health_check/embeddings_health_check_mixin.py +31 -0
  707. rasa/shared/utils/health_check/health_check.py +258 -0
  708. rasa/shared/utils/health_check/llm_health_check_mixin.py +31 -0
  709. rasa/shared/utils/io.py +499 -0
  710. rasa/shared/utils/llm.py +760 -0
  711. rasa/shared/utils/pykwalify_extensions.py +27 -0
  712. rasa/shared/utils/schemas/__init__.py +0 -0
  713. rasa/shared/utils/schemas/config.yml +2 -0
  714. rasa/shared/utils/schemas/domain.yml +145 -0
  715. rasa/shared/utils/schemas/events.py +214 -0
  716. rasa/shared/utils/schemas/model_config.yml +36 -0
  717. rasa/shared/utils/schemas/stories.yml +173 -0
  718. rasa/shared/utils/yaml.py +1067 -0
  719. rasa/studio/__init__.py +0 -0
  720. rasa/studio/auth.py +270 -0
  721. rasa/studio/config.py +136 -0
  722. rasa/studio/constants.py +19 -0
  723. rasa/studio/data_handler.py +368 -0
  724. rasa/studio/download.py +489 -0
  725. rasa/studio/results_logger.py +137 -0
  726. rasa/studio/train.py +134 -0
  727. rasa/studio/upload.py +549 -0
  728. rasa/telemetry.py +1869 -0
  729. rasa/tracing/__init__.py +0 -0
  730. rasa/tracing/config.py +355 -0
  731. rasa/tracing/constants.py +62 -0
  732. rasa/tracing/instrumentation/__init__.py +0 -0
  733. rasa/tracing/instrumentation/attribute_extractors.py +764 -0
  734. rasa/tracing/instrumentation/instrumentation.py +1306 -0
  735. rasa/tracing/instrumentation/intentless_policy_instrumentation.py +144 -0
  736. rasa/tracing/instrumentation/metrics.py +294 -0
  737. rasa/tracing/metric_instrument_provider.py +205 -0
  738. rasa/utils/__init__.py +0 -0
  739. rasa/utils/beta.py +83 -0
  740. rasa/utils/cli.py +28 -0
  741. rasa/utils/common.py +639 -0
  742. rasa/utils/converter.py +53 -0
  743. rasa/utils/endpoints.py +331 -0
  744. rasa/utils/io.py +252 -0
  745. rasa/utils/json_utils.py +60 -0
  746. rasa/utils/licensing.py +542 -0
  747. rasa/utils/log_utils.py +181 -0
  748. rasa/utils/mapper.py +210 -0
  749. rasa/utils/ml_utils.py +147 -0
  750. rasa/utils/plotting.py +362 -0
  751. rasa/utils/sanic_error_handler.py +32 -0
  752. rasa/utils/singleton.py +23 -0
  753. rasa/utils/tensorflow/__init__.py +0 -0
  754. rasa/utils/tensorflow/callback.py +112 -0
  755. rasa/utils/tensorflow/constants.py +116 -0
  756. rasa/utils/tensorflow/crf.py +492 -0
  757. rasa/utils/tensorflow/data_generator.py +440 -0
  758. rasa/utils/tensorflow/environment.py +161 -0
  759. rasa/utils/tensorflow/exceptions.py +5 -0
  760. rasa/utils/tensorflow/feature_array.py +366 -0
  761. rasa/utils/tensorflow/layers.py +1565 -0
  762. rasa/utils/tensorflow/layers_utils.py +113 -0
  763. rasa/utils/tensorflow/metrics.py +281 -0
  764. rasa/utils/tensorflow/model_data.py +798 -0
  765. rasa/utils/tensorflow/model_data_utils.py +499 -0
  766. rasa/utils/tensorflow/models.py +935 -0
  767. rasa/utils/tensorflow/rasa_layers.py +1094 -0
  768. rasa/utils/tensorflow/transformer.py +640 -0
  769. rasa/utils/tensorflow/types.py +6 -0
  770. rasa/utils/train_utils.py +572 -0
  771. rasa/utils/url_tools.py +53 -0
  772. rasa/utils/yaml.py +54 -0
  773. rasa/validator.py +1653 -0
  774. rasa/version.py +3 -0
  775. rasa_pro-3.11.3.dist-info/METADATA +198 -0
  776. rasa_pro-3.11.3.dist-info/NOTICE +5 -0
  777. rasa_pro-3.11.3.dist-info/RECORD +779 -0
  778. rasa_pro-3.11.3.dist-info/WHEEL +4 -0
  779. rasa_pro-3.11.3.dist-info/entry_points.txt +3 -0
rasa/telemetry.py ADDED
@@ -0,0 +1,1869 @@
1
+ import asyncio
2
+ import contextlib
3
+ import hashlib
4
+ import inspect
5
+ import json
6
+ import logging
7
+ import multiprocessing
8
+ import os
9
+ import platform
10
+ import sys
11
+ import textwrap
12
+ import typing
13
+ import uuid
14
+ from collections import defaultdict
15
+ from datetime import datetime
16
+ from functools import wraps
17
+ from pathlib import Path
18
+ from typing import Any, Callable, Dict, List, Optional, Text
19
+
20
+ import importlib_resources
21
+ import requests
22
+ from terminaltables import SingleTable
23
+
24
+ import rasa
25
+ import rasa.anonymization.utils
26
+ import rasa.shared.utils.io
27
+ import rasa.utils.io
28
+ from rasa import model
29
+ from rasa.constants import (
30
+ CONFIG_FILE_TELEMETRY_KEY,
31
+ CONFIG_TELEMETRY_DATE,
32
+ CONFIG_TELEMETRY_ENABLED,
33
+ CONFIG_TELEMETRY_ID,
34
+ )
35
+ from rasa.shared.constants import (
36
+ PROMPT_CONFIG_KEY,
37
+ PROMPT_TEMPLATE_CONFIG_KEY,
38
+ LLM_API_HEALTH_CHECK_ENV_VAR,
39
+ LLM_API_HEALTH_CHECK_DEFAULT_VALUE,
40
+ MODEL_GROUP_CONFIG_KEY,
41
+ )
42
+ from rasa.engine.storage.local_model_storage import LocalModelStorage
43
+ from rasa.shared.constants import DOCS_URL_TELEMETRY, UTTER_ASK_PREFIX
44
+ from rasa.shared.core.flows import Flow
45
+ from rasa.shared.core.flows.steps import (
46
+ CollectInformationFlowStep,
47
+ SetSlotsFlowStep,
48
+ LinkFlowStep,
49
+ CallFlowStep,
50
+ )
51
+ from rasa.shared.exceptions import RasaException
52
+ from rasa.utils import common as rasa_utils
53
+
54
+ if typing.TYPE_CHECKING:
55
+ from rasa.core.brokers.broker import EventBroker
56
+ from rasa.core.tracker_store import TrackerStore
57
+ from rasa.core.channels.channel import InputChannel
58
+ from rasa.core.agent import Agent
59
+ from rasa.shared.nlu.training_data.training_data import TrainingData
60
+ from rasa.shared.importers.importer import TrainingDataImporter
61
+ from rasa.core.utils import AvailableEndpoints
62
+ from rasa.e2e_test.e2e_test_case import TestCase, Fixture, Metadata
63
+
64
+ logger = logging.getLogger(__name__)
65
+
66
+ SEGMENT_TRACK_ENDPOINT = "https://api.segment.io/v1/track"
67
+ SEGMENT_IDENTIFY_ENDPOINT = "https://api.segment.io/v1/identify"
68
+ SEGMENT_REQUEST_TIMEOUT = 5 # seconds
69
+
70
+ TELEMETRY_ENABLED_ENVIRONMENT_VARIABLE = "RASA_TELEMETRY_ENABLED"
71
+ TELEMETRY_DEBUG_ENVIRONMENT_VARIABLE = "RASA_TELEMETRY_DEBUG"
72
+ RASA_PRO_CONFIG_FILE_TELEMETRY_KEY = "traits"
73
+
74
+ # the environment variable can be used for local development to set a test key
75
+ # e.g. `RASA_TELEMETRY_WRITE_KEY=12354 rasa train`
76
+ TELEMETRY_WRITE_KEY_ENVIRONMENT_VARIABLE = "RASA_TELEMETRY_WRITE_KEY"
77
+ EXCEPTION_WRITE_KEY_ENVIRONMENT_VARIABLE = "RASA_EXCEPTION_WRITE_KEY"
78
+
79
+ TELEMETRY_ID = "metrics_id"
80
+ TELEMETRY_ENABLED_BY_DEFAULT = True
81
+
82
+ # if one of these environment variables is set, we assume to be running in CI env
83
+ CI_ENVIRONMENT_TELL = [
84
+ "bamboo.buildKey",
85
+ "BUILD_ID",
86
+ "BUILD_NUMBER",
87
+ "BUILDKITE",
88
+ "CI",
89
+ "CIRCLECI",
90
+ "CONTINUOUS_INTEGRATION",
91
+ "GITHUB_ACTIONS",
92
+ "HUDSON_URL",
93
+ "JENKINS_URL",
94
+ "TEAMCITY_VERSION",
95
+ "TRAVIS",
96
+ "CODEBUILD_BUILD_ARN",
97
+ "CODEBUILD_BUILD_ID",
98
+ "CODEBUILD_BATCH_BUILD_IDENTIFIER",
99
+ ]
100
+
101
+ # If updating or creating a new event, remember to update
102
+ # https://rasa.com/docs/rasa-pro/telemetry/telemetry OR
103
+ # https://rasa.com/docs/rasa-pro/telemetry/reference
104
+ TRAINING_STARTED_EVENT = "Training Started"
105
+ TRAINING_COMPLETED_EVENT = "Training Completed"
106
+ TELEMETRY_DISABLED_EVENT = "Telemetry Disabled"
107
+ TELEMETRY_DATA_SPLIT_EVENT = "Training Data Split"
108
+ TELEMETRY_DATA_VALIDATED_EVENT = "Training Data Validated"
109
+ TELEMETRY_DATA_CONVERTED_EVENT = "Training Data Converted"
110
+ TELEMETRY_TRACKER_EXPORTED_EVENT = "Tracker Exported"
111
+ TELEMETRY_INTERACTIVE_LEARNING_STARTED_EVENT = "Interactive Learning Started"
112
+ TELEMETRY_SERVER_STARTED_EVENT = "Server Started"
113
+ TELEMETRY_PROJECT_CREATED_EVENT = "Project Created"
114
+ TELEMETRY_SHELL_STARTED_EVENT = "Shell Started"
115
+ TELEMETRY_INSPECT_STARTED_EVENT = "Inspect Started"
116
+ TELEMETRY_VISUALIZATION_STARTED_EVENT = "Story Visualization Started"
117
+ TELEMETRY_TEST_CORE_EVENT = "Model Core Tested"
118
+ TELEMETRY_TEST_NLU_EVENT = "Model NLU Tested"
119
+ TELEMETRY_MARKERS_EXTRACTION_INITIATED_EVENT = "Markers Extraction Initiated"
120
+ TELEMETRY_MARKERS_EXTRACTED_EVENT = "Markers Extracted"
121
+ TELEMETRY_MARKERS_STATS_COMPUTED_EVENT = "Markers Statistics Computed"
122
+ TELEMETRY_MARKERS_PARSED_COUNT = "Markers Parsed"
123
+
124
+ TELEMETRY_RESPONSE_REPHRASED_EVENT = "Response Rephrased"
125
+ TELEMETRY_INTENTLESS_POLICY_TRAINING_STARTED_EVENT = (
126
+ "Intentless Policy Training Started"
127
+ )
128
+ TELEMETRY_INTENTLESS_POLICY_TRAINING_COMPLETED_EVENT = (
129
+ "Intentless Policy Training Completed"
130
+ )
131
+ TELEMETRY_INTENTLESS_POLICY_PREDICT_EVENT = "Intentless Policy Predicted"
132
+ TELEMETRY_E2E_TEST_RUN_STARTED_EVENT = "E2E Test Run Started"
133
+ TELEMETRY_ENTERPRISE_SEARCH_POLICY_TRAINING_STARTED_EVENT = (
134
+ "Enterprise Search Policy Training Started"
135
+ )
136
+ TELEMETRY_ENTERPRISE_SEARCH_POLICY_TRAINING_COMPLETED_EVENT = (
137
+ "Enterprise Search Policy Training Completed"
138
+ )
139
+ TELEMETRY_ENTERPRISE_SEARCH_POLICY_PREDICT_EVENT = "Enterprise Search Policy Predicted"
140
+
141
+ # licensing events
142
+ TELEMETRY_CONVERSATION_COUNT = "Conversation Count"
143
+ TELEMETRY_CONVERSATION_SOFT_LIMIT_REACHED = "Conversation Soft Limit Reached"
144
+ TELEMETRY_CONVERSATION_HARD_LIMIT_REACHED = "Conversation Hard Limit Reached"
145
+
146
+ # used to calculate the context on the first call and cache it afterwards
147
+ TELEMETRY_CONTEXT = None
148
+
149
+ # constants used for the training started events
150
+ NUM_FLOWS = "num_flows"
151
+ NUM_FLOWS_WITH_NLU_TRIGGER = "num_flows_with_nlu_trigger"
152
+ NUM_FLOWS_WITH_FLOW_GUARDS = "num_flows_with_flow_guards"
153
+ NUM_FLOWS_ALWAYS_INCLUDED_IN_PROMPT = "num_flows_always_included_in_prompt"
154
+ NUM_FLOWS_WITH_NOT_STARTABLE_FLOW_GUARDS = "num_flows_with_not_startable_flow_guards"
155
+ NUM_COLLECT_STEPS = "num_collect_steps"
156
+ NUM_COLLECT_STEPS_WITH_SEPARATE_UTTER = "num_collect_steps_with_separate_utter"
157
+ NUM_COLLECT_STEPS_WITH_REJECTIONS = "num_collect_steps_with_rejections"
158
+ NUM_COLLECT_STEPS_WITH_NOT_RESET_AFTER_FLOW_ENDS = (
159
+ "num_collect_steps_with_not_reset_after_flow_ends"
160
+ )
161
+ NUM_SET_SLOT_STEPS = "num_set_slot_steps"
162
+ MAX_DEPTH_OF_IF_CONSTRUCT = "max_depth_of_if_construct"
163
+ NUM_LINK_STEPS = "num_link_steps"
164
+ NUM_CALL_STEPS = "num_call_steps"
165
+ NUM_SHARED_SLOTS_BETWEEN_FLOWS = "num_shared_slots_between_flows"
166
+ LLM_COMMAND_GENERATOR_MODEL_NAME = "llm_command_generator_model_name"
167
+ LLM_COMMAND_GENERATOR_MODEL_GROUP_ID = "llm_command_generator_model_group_id"
168
+ LLM_COMMAND_GENERATOR_CUSTOM_PROMPT_USED = "llm_command_generator_custom_prompt_used"
169
+ MULTI_STEP_LLM_COMMAND_GENERATOR_HANDLE_FLOWS_PROMPT_USED = (
170
+ "multi_step_llm_command_generator_custom_handle_flows_prompt_used"
171
+ )
172
+ MULTI_STEP_LLM_COMMAND_GENERATOR_FILL_SLOTS_PROMPT_USED = (
173
+ "multi_step_llm_command_generator_custom_fill_slots_prompt_used"
174
+ )
175
+ FLOW_RETRIEVAL_ENABLED = "flow_retrieval_enabled"
176
+ FLOW_RETRIEVAL_EMBEDDING_MODEL_NAME = "flow_retrieval_embedding_model_name"
177
+ FLOW_RETRIEVAL_EMBEDDING_MODEL_GROUP_ID = "flow_retrieval_embedding_model_group_id"
178
+ TRACING_BACKEND = "tracing_backend"
179
+ METRICS_BACKEND = "metrics_backend"
180
+ VERSION = "version"
181
+
182
+ # E2E test conversion
183
+ TELEMETRY_E2E_TEST_CONVERSION_EVENT = "E2E Test Conversion Completed"
184
+ E2E_TEST_CONVERSION_FILE_TYPE = "file_type"
185
+ E2E_TEST_CONVERSION_TEST_CASE_COUNT = "test_case_count"
186
+
187
+
188
+ def print_telemetry_reporting_info() -> None:
189
+ """Print telemetry information to std out."""
190
+ message = textwrap.dedent(
191
+ f"""
192
+ Rasa Pro reports anonymous usage telemetry to help improve the product
193
+ for all its users.
194
+
195
+ If you'd like to opt-out, you can use `rasa telemetry disable`.
196
+ To learn more, check out {DOCS_URL_TELEMETRY}."""
197
+ ).strip()
198
+
199
+ table = SingleTable([[message]])
200
+ print(table.table)
201
+
202
+
203
+ def _default_telemetry_configuration(is_enabled: bool) -> Dict[Text, Any]:
204
+ return {
205
+ CONFIG_TELEMETRY_ENABLED: is_enabled,
206
+ CONFIG_TELEMETRY_ID: uuid.uuid4().hex,
207
+ CONFIG_TELEMETRY_DATE: datetime.now(),
208
+ }
209
+
210
+
211
+ def _write_default_telemetry_configuration(
212
+ is_enabled: bool = TELEMETRY_ENABLED_BY_DEFAULT,
213
+ ) -> bool:
214
+ new_config = _default_telemetry_configuration(is_enabled)
215
+
216
+ keys = [CONFIG_FILE_TELEMETRY_KEY, RASA_PRO_CONFIG_FILE_TELEMETRY_KEY]
217
+
218
+ success = all(
219
+ [rasa_utils.write_global_config_value(key, new_config) for key in keys]
220
+ )
221
+
222
+ # Do not show info if user has enabled/disabled telemetry via env var
223
+ telemetry_environ = os.environ.get(TELEMETRY_ENABLED_ENVIRONMENT_VARIABLE)
224
+ if is_enabled and success and telemetry_environ is None:
225
+ print_telemetry_reporting_info()
226
+
227
+ return success
228
+
229
+
230
+ def _is_telemetry_enabled_in_configuration() -> bool:
231
+ """Read telemetry configuration from the user's Rasa config file in $HOME.
232
+
233
+ Creates a default configuration if no configuration exists.
234
+
235
+ Returns:
236
+ `True`, if telemetry is enabled, `False` otherwise.
237
+ """
238
+ try:
239
+ stored_config = rasa_utils.read_global_config_value(
240
+ CONFIG_FILE_TELEMETRY_KEY, unavailable_ok=False
241
+ )
242
+
243
+ return stored_config[CONFIG_TELEMETRY_ENABLED]
244
+ except ValueError as e:
245
+ logger.debug(f"Could not read telemetry settings from configuration file: {e}")
246
+
247
+ # seems like there is no config, we'll create one and enable telemetry
248
+ success = _write_default_telemetry_configuration()
249
+ # if writing the configuration failed, telemetry will be disabled
250
+ return TELEMETRY_ENABLED_BY_DEFAULT and success
251
+
252
+
253
+ def is_telemetry_enabled() -> bool:
254
+ """Check if telemetry is enabled either in configuration or environment.
255
+
256
+ Returns:
257
+ `True`, if telemetry is enabled, `False` otherwise.
258
+ """
259
+ from rasa.utils import licensing
260
+
261
+ if licensing.is_champion_server_license():
262
+ logger.debug("Telemetry is enabled for developer licenses.")
263
+ return True
264
+
265
+ telemetry_environ = os.environ.get(TELEMETRY_ENABLED_ENVIRONMENT_VARIABLE)
266
+
267
+ if telemetry_environ is not None:
268
+ return telemetry_environ.lower() == "true"
269
+
270
+ try:
271
+ return rasa_utils.read_global_config_value(
272
+ CONFIG_FILE_TELEMETRY_KEY, unavailable_ok=False
273
+ )[CONFIG_TELEMETRY_ENABLED]
274
+ except ValueError:
275
+ return False
276
+
277
+
278
+ def initialize_telemetry() -> bool:
279
+ """Read telemetry configuration from the user's Rasa config file in $HOME.
280
+
281
+ Creates a default configuration if no configuration exists.
282
+
283
+ Returns:
284
+ `True`, if telemetry is enabled, `False` otherwise.
285
+ """
286
+ try:
287
+ # calling this even if the environment variable is set makes sure the
288
+ # configuration is created and there is a telemetry ID
289
+ is_enabled_in_configuration = _is_telemetry_enabled_in_configuration()
290
+
291
+ telemetry_environ = os.environ.get(TELEMETRY_ENABLED_ENVIRONMENT_VARIABLE)
292
+
293
+ if telemetry_environ is None:
294
+ return is_enabled_in_configuration
295
+
296
+ return telemetry_environ.lower() == "true"
297
+ except Exception as e: # skipcq:PYL-W0703
298
+ logger.exception(
299
+ f"Failed to initialize telemetry reporting: {e}."
300
+ f"Telemetry reporting will be disabled."
301
+ )
302
+ return False
303
+
304
+
305
+ def ensure_telemetry_enabled(f: Callable[..., Any]) -> Callable[..., Any]:
306
+ """Function decorator for telemetry functions that ensures telemetry is enabled.
307
+
308
+ WARNING: does not work as a decorator for async generators.
309
+
310
+ Args:
311
+ f: function to call if telemetry is enabled
312
+ Returns:
313
+ Return wrapped function
314
+ """
315
+ # allows us to use the decorator for async generator functions
316
+ if inspect.isasyncgenfunction(f):
317
+
318
+ @wraps(f)
319
+ async def decorated_async_gen(*args: Any, **kwargs: Any) -> Any:
320
+ if is_telemetry_enabled():
321
+ yield f(*args, **kwargs)
322
+
323
+ return decorated_async_gen
324
+
325
+ # allows us to use the decorator for async and non async functions
326
+ if asyncio.iscoroutinefunction(f):
327
+
328
+ @wraps(f)
329
+ async def decorated_coroutine(*args: Any, **kwargs: Any) -> Any:
330
+ if is_telemetry_enabled():
331
+ return await f(*args, **kwargs)
332
+ return None
333
+
334
+ return decorated_coroutine
335
+
336
+ @wraps(f)
337
+ def decorated(*args: Any, **kwargs: Any) -> Any:
338
+ if is_telemetry_enabled():
339
+ return f(*args, **kwargs)
340
+ return None
341
+
342
+ return decorated
343
+
344
+
345
+ def _fetch_write_key(tool: Text, environment_variable: Text) -> Optional[Text]:
346
+ """Read the write key from a tool from our set of keys.
347
+
348
+ Args:
349
+ tool: name of the tool we want to fetch a key for
350
+ environment_variable: name of the environment variable to set the key
351
+ Returns:
352
+ write key, if a key was present.
353
+ """
354
+ import importlib_resources
355
+ from rasa import __name__ as name
356
+
357
+ if os.environ.get(environment_variable):
358
+ # a write key set using the environment variable will always
359
+ # overwrite any key provided as part of the package (`keys` file)
360
+ return os.environ.get(environment_variable)
361
+
362
+ write_key_path = str(importlib_resources.files(name).joinpath("keys"))
363
+
364
+ # noinspection PyBroadException
365
+ try:
366
+ with open(write_key_path) as f:
367
+ return json.load(f).get(tool)
368
+ except Exception: # skipcq:PYL-W0703
369
+ return None
370
+
371
+
372
+ def telemetry_write_key() -> Optional[Text]:
373
+ """Read the Segment write key from the segment key text file.
374
+
375
+ The segment key text file should by present only in wheel/sdist packaged
376
+ versions of Rasa Pro. This avoids running telemetry locally when
377
+ developing on Rasa or when running CI builds.
378
+
379
+ In local development, this should always return `None` to avoid logging telemetry.
380
+
381
+ Returns:
382
+ Segment write key, if the key file was present.
383
+ """
384
+ return _fetch_write_key("segment", TELEMETRY_WRITE_KEY_ENVIRONMENT_VARIABLE)
385
+
386
+
387
+ def sentry_write_key() -> Optional[Text]:
388
+ """Read the sentry write key from the sentry key text file.
389
+
390
+ Returns:
391
+ Sentry write key, if the key file was present.
392
+ """
393
+ return _fetch_write_key("sentry", EXCEPTION_WRITE_KEY_ENVIRONMENT_VARIABLE)
394
+
395
+
396
+ def _encode_base64(original: Text, encoding: Text = "utf-8") -> Text:
397
+ """Encodes a string as a base64 string.
398
+
399
+ Args:
400
+ original: Text to be encoded.
401
+ encoding: Encoding used to convert text to binary.
402
+
403
+ Returns:
404
+ Encoded text.
405
+ """
406
+ import base64
407
+
408
+ return base64.b64encode(original.encode(encoding)).decode(encoding)
409
+
410
+
411
+ def segment_request_header(write_key: Text) -> Dict[Text, Any]:
412
+ """Use a segment write key to create authentication headers for the segment API.
413
+
414
+ Args:
415
+ write_key: Authentication key for segment.
416
+
417
+ Returns:
418
+ Authentication headers for segment.
419
+ """
420
+ return {
421
+ "Authorization": "Basic {}".format(_encode_base64(write_key + ":")),
422
+ "Content-Type": "application/json",
423
+ }
424
+
425
+
426
+ def segment_request_payload(
427
+ distinct_id: Text,
428
+ event_name: Text,
429
+ properties: Dict[Text, Any],
430
+ context: Dict[Text, Any],
431
+ ) -> Dict[Text, Any]:
432
+ """Compose a valid payload for the segment API.
433
+
434
+ Args:
435
+ distinct_id: Unique telemetry ID.
436
+ event_name: Name of the event.
437
+ properties: Values to report along the event.
438
+ context: Context information about the event.
439
+
440
+ Returns:
441
+ Valid segment payload.
442
+ """
443
+ return {
444
+ "userId": distinct_id,
445
+ "event": event_name,
446
+ "properties": properties,
447
+ "context": context,
448
+ }
449
+
450
+
451
+ def in_continuous_integration() -> bool:
452
+ """Returns `True` if currently running inside a continuous integration context."""
453
+ return any(env in os.environ for env in CI_ENVIRONMENT_TELL)
454
+
455
+
456
+ def _is_telemetry_debug_enabled() -> bool:
457
+ """Check if telemetry debug mode is enabled."""
458
+ return (
459
+ os.environ.get(TELEMETRY_DEBUG_ENVIRONMENT_VARIABLE, "false").lower() == "true"
460
+ )
461
+
462
+
463
+ def print_telemetry_payload(payload: Dict[Text, Any]) -> None:
464
+ """Print a telemetry payload to the commandline.
465
+
466
+ Args:
467
+ payload: payload to be delivered to segment.
468
+ """
469
+ payload_json = json.dumps(payload, indent=2)
470
+ logger.debug(f"Telemetry payload: {payload_json}")
471
+
472
+
473
+ def _get_telemetry_write_key() -> Optional[Text]:
474
+ if os.environ.get(TELEMETRY_WRITE_KEY_ENVIRONMENT_VARIABLE):
475
+ return os.environ.get(TELEMETRY_WRITE_KEY_ENVIRONMENT_VARIABLE)
476
+
477
+ write_key_path = str(importlib_resources.files(rasa.__name__).joinpath("keys"))
478
+
479
+ try:
480
+ with open(write_key_path) as f:
481
+ return json.load(f).get("segment")
482
+ except Exception:
483
+ return None
484
+
485
+
486
+ def _send_event(
487
+ distinct_id: Text,
488
+ event_name: Text,
489
+ properties: Dict[Text, Any],
490
+ context: Dict[Text, Any],
491
+ ) -> None:
492
+ """Report the contents segmentof an event to the /track Segment endpoint.
493
+
494
+ Documentation: https://.com/docs/sources/server/http/
495
+
496
+ Do not call this function from outside telemetry.py! This function does not
497
+ check if telemetry is enabled or not.
498
+
499
+ Args:
500
+ distinct_id: Unique telemetry ID.
501
+ event_name: Name of the event.
502
+ properties: Values to report along the event.
503
+ context: Context information about the event.
504
+ """
505
+ payload = segment_request_payload(distinct_id, event_name, properties, context)
506
+
507
+ _send_request(SEGMENT_TRACK_ENDPOINT, payload)
508
+
509
+
510
+ def _send_request(url: Text, payload: Dict[Text, Any]) -> None:
511
+ """Send a request to the Segment API.
512
+
513
+ Args:
514
+ url: URL of the Segment API endpoint
515
+ payload: payload to send to the Segment API
516
+ """
517
+ if _is_telemetry_debug_enabled():
518
+ print_telemetry_payload(payload)
519
+ return
520
+
521
+ write_key = _get_telemetry_write_key()
522
+ if not write_key:
523
+ # If RASA_TELEMETRY_WRITE_KEY is empty or `None`, telemetry has not
524
+ # been enabled for this build (e.g. because it is running from source)
525
+ logger.debug("Skipping request to external service: telemetry key not set.")
526
+ return
527
+
528
+ headers = rasa.telemetry.segment_request_header(write_key)
529
+
530
+ resp = requests.post(
531
+ url=url,
532
+ headers=headers,
533
+ json=payload,
534
+ timeout=SEGMENT_REQUEST_TIMEOUT,
535
+ )
536
+ # handle different failure cases
537
+ if resp.status_code != 200:
538
+ logger.debug(
539
+ f"Segment telemetry request returned a {resp.status_code} response. "
540
+ f"Body: {resp.text}"
541
+ )
542
+ else:
543
+ data = resp.json()
544
+ if not data.get("success"):
545
+ logger.debug(
546
+ f"Segment telemetry request returned a failure. Response: {data}"
547
+ )
548
+
549
+
550
+ def _hash_directory_path(path: Text) -> Optional[Text]:
551
+ """Create a hash for the directory.
552
+
553
+ Returns:
554
+ hash of the directories path
555
+ """
556
+ full_path = Path(path).absolute()
557
+ return hashlib.sha256(str(full_path).encode("utf-8")).hexdigest()
558
+
559
+
560
+ # noinspection PyBroadException
561
+ def _is_docker() -> bool:
562
+ """Guess if we are running in docker environment.
563
+
564
+ Returns:
565
+ `True` if we are running inside docker, `False` otherwise.
566
+ """
567
+ # first we try to use the env
568
+ try:
569
+ os.stat("/.dockerenv")
570
+ return True
571
+ except Exception: # skipcq:PYL-W0703
572
+ pass
573
+
574
+ # if that didn't work, try to use proc information
575
+ try:
576
+ return "docker" in rasa.shared.utils.io.read_file("/proc/self/cgroup", "utf8")
577
+ except Exception: # skipcq:PYL-W0703
578
+ return False
579
+
580
+
581
+ def with_default_context_fields(
582
+ context: Optional[Dict[Text, Any]] = None,
583
+ ) -> Dict[Text, Any]:
584
+ """Return a new context dictionary with default and provided field values merged.
585
+
586
+ The default fields contain only the OS information for now.
587
+
588
+ Args:
589
+ context: Context information about the event.
590
+
591
+ Return:
592
+ A new context.
593
+ """
594
+ context = context or {}
595
+
596
+ return {**_default_context_fields(), **context}
597
+
598
+
599
+ def _default_context_fields() -> Dict[Text, Any]:
600
+ """Return a dictionary that contains the default context values.
601
+
602
+ Return:
603
+ A new context containing information about the runtime environment.
604
+ """
605
+ from rasa.utils.licensing import property_of_active_license, get_license_hash
606
+
607
+ global TELEMETRY_CONTEXT
608
+
609
+ if not TELEMETRY_CONTEXT:
610
+ # Make sure to update the example in docs/docs/telemetry/telemetry.mdx
611
+ # if you change / add context
612
+ TELEMETRY_CONTEXT = {
613
+ "os": {"name": platform.system(), "version": platform.release()},
614
+ "ci": in_continuous_integration(),
615
+ "project": model.project_fingerprint(),
616
+ "directory": _hash_directory_path(os.getcwd()),
617
+ "python": sys.version.split(" ")[0],
618
+ "rasa_pro": rasa.__version__,
619
+ "cpu": multiprocessing.cpu_count(),
620
+ "docker": _is_docker(),
621
+ "license_hash": get_license_hash(),
622
+ "company": property_of_active_license(
623
+ lambda active_license: active_license.company
624
+ ),
625
+ }
626
+
627
+ # avoid returning the cached dict --> caller could modify the dictionary...
628
+ # usually we would use `lru_cache`, but that doesn't return a dict copy and
629
+ # doesn't work on inner functions, so we need to roll our own caching...
630
+ return TELEMETRY_CONTEXT.copy()
631
+
632
+
633
+ def _track(
634
+ event_name: Text,
635
+ properties: Optional[Dict[Text, Any]] = None,
636
+ context: Optional[Dict[Text, Any]] = None,
637
+ ) -> None:
638
+ """Tracks a telemetry event.
639
+
640
+ It is OK to use this function from outside telemetry.py, but note that it
641
+ is recommended to create a new track_xyz() function for complex telemetry
642
+ events, or events that are generated from many parts of the Rasa Pro code.
643
+
644
+ Args:
645
+ event_name: Name of the event.
646
+ properties: Dictionary containing the event's properties.
647
+ context: Dictionary containing some context for this event.
648
+ """
649
+ try:
650
+ telemetry_id = get_telemetry_id()
651
+
652
+ if not telemetry_id:
653
+ logger.debug("Will not report telemetry events as no ID was found.")
654
+ return
655
+
656
+ if not properties:
657
+ properties = {}
658
+
659
+ properties[TELEMETRY_ID] = telemetry_id
660
+
661
+ # this is an additional check in case _track() is called
662
+ # from a function that is not decorated with @ensure_telemetry_enabled
663
+ if is_telemetry_enabled():
664
+ _send_event(
665
+ telemetry_id,
666
+ event_name,
667
+ properties,
668
+ with_default_context_fields(context),
669
+ )
670
+ except Exception as e: # skipcq:PYL-W0703
671
+ logger.debug(f"Skipping telemetry reporting: {e}")
672
+
673
+
674
+ def _identify(
675
+ traits: Optional[Dict[Text, Any]] = None,
676
+ context: Optional[Dict[Text, Any]] = None,
677
+ ) -> None:
678
+ """Tracks telemetry traits.
679
+
680
+ It is OK to use this function from outside telemetry.py, but note that it
681
+ is recommended to create a new track_xyz() function for complex telemetry
682
+ traits, or traits that are generated from many parts of the Rasa Pro code.
683
+
684
+ Args:
685
+ traits: Dictionary containing the tracked traits.
686
+ context: Dictionary containing some context for the traits.
687
+ """
688
+ try:
689
+ telemetry_id = get_telemetry_id()
690
+
691
+ if not telemetry_id:
692
+ logger.debug("Will not report telemetry events as no ID was found.")
693
+ return
694
+
695
+ if not traits:
696
+ traits = {}
697
+
698
+ _send_traits(telemetry_id, traits, with_default_context_fields(context))
699
+ except Exception as e:
700
+ logger.debug(f"Skipping telemetry reporting: {e}")
701
+
702
+
703
+ def _send_traits(
704
+ distinct_id: Text,
705
+ traits: Dict[Text, Any],
706
+ context: Dict[Text, Any],
707
+ ) -> None:
708
+ """Report the contents of telemetry traits to the /identify Segment endpoint.
709
+
710
+ Do not call this function from outside telemetry.py! This function does not
711
+ check if telemetry is enabled or not.
712
+
713
+ Args:
714
+ distinct_id: Unique telemetry ID.
715
+ traits: Pieces of information to be recorded about
716
+ rasa_plus interface implementations.
717
+ context: Context information to be sent along with traits.
718
+ """
719
+ payload = segment_identify_request_payload(distinct_id, traits, context)
720
+
721
+ _send_request(SEGMENT_IDENTIFY_ENDPOINT, payload)
722
+
723
+
724
+ def segment_identify_request_payload(
725
+ distinct_id: Text,
726
+ traits: Dict[Text, Any],
727
+ context: Dict[Text, Any],
728
+ ) -> Dict[Text, Any]:
729
+ """Compose a valid payload for the segment API.
730
+
731
+ Args:
732
+ distinct_id: Unique telemetry ID.
733
+ traits: Pieces of information to be recorded about
734
+ rasa_plus interface implementations.
735
+ context: Context information to be sent along with traits.
736
+
737
+ Returns:
738
+ Valid segment payload.
739
+ """
740
+ return {
741
+ "userId": distinct_id,
742
+ "traits": traits,
743
+ "context": context,
744
+ }
745
+
746
+
747
+ def get_telemetry_id() -> Optional[Text]:
748
+ """Return the unique telemetry identifier for this Rasa Pro install.
749
+
750
+ The identifier can be based on the license.
751
+ Otherwise, it can be any string, but it should be a UUID.
752
+
753
+ Returns:
754
+ The identifier, if it is configured correctly.
755
+ """
756
+ from rasa.utils.licensing import property_of_active_license
757
+
758
+ return property_of_active_license(lambda active_license: active_license.jti)
759
+
760
+
761
+ def toggle_telemetry_reporting(is_enabled: bool) -> None:
762
+ """Write to the configuration if telemetry tracking should be enabled or disabled.
763
+
764
+ Args:
765
+ is_enabled: `True` if the telemetry reporting should be enabled,
766
+ `False` otherwise.
767
+ """
768
+ configuration = rasa_utils.read_global_config_value(CONFIG_FILE_TELEMETRY_KEY)
769
+
770
+ if configuration:
771
+ configuration[CONFIG_TELEMETRY_ENABLED] = is_enabled
772
+ else:
773
+ configuration = _default_telemetry_configuration(is_enabled)
774
+
775
+ rasa_utils.write_global_config_value(CONFIG_FILE_TELEMETRY_KEY, configuration)
776
+ rasa_utils.write_global_config_value(
777
+ RASA_PRO_CONFIG_FILE_TELEMETRY_KEY, configuration
778
+ )
779
+
780
+
781
+ def filter_errors(
782
+ event: Optional[Dict[Text, Any]], hint: Optional[Dict[Text, Any]] = None
783
+ ) -> Optional[Dict[Text, Any]]:
784
+ """Filter errors.
785
+
786
+ Args:
787
+ event: event to be logged to sentry
788
+ hint: some hinting information sent alongside of the event
789
+
790
+ Returns:
791
+ the event without any sensitive / PII data or `None` if the event constitutes
792
+ an `ImportError` which should be discarded.
793
+ """
794
+ if hint and "exc_info" in hint:
795
+ exc_type, exc_value, tb = hint["exc_info"]
796
+ if isinstance(exc_value, ImportError):
797
+ return None
798
+ return event
799
+
800
+
801
+ def before_send(
802
+ event: Dict[Text, Any], _unused_hint: Optional[Dict[Text, Any]] = None
803
+ ) -> Optional[Dict[Text, Any]]:
804
+ """Strips the sensitive data and filters errors before sending to sentry.
805
+
806
+ Args:
807
+ event: event to be logged to sentry
808
+ _unused_hint: some hinting information sent alongside of the event
809
+
810
+ Returns:
811
+ the event without any sensitive / PII data or `None` if the event should
812
+ be discarded.
813
+ """
814
+ cleaned_event = strip_sensitive_data_from_sentry_event(event, _unused_hint)
815
+ return filter_errors(cleaned_event, _unused_hint)
816
+
817
+
818
+ def strip_sensitive_data_from_sentry_event(
819
+ event: Dict[Text, Any], _unused_hint: Optional[Dict[Text, Any]] = None
820
+ ) -> Optional[Dict[Text, Any]]:
821
+ """Remove any sensitive data from the event (e.g. path names).
822
+
823
+ Args:
824
+ event: event to be logged to sentry
825
+ _unused_hint: some hinting information sent alongside of the event
826
+
827
+ Returns:
828
+ the event without any sensitive / PII data or `None` if the event should
829
+ be discarded.
830
+ """
831
+ # removes any paths from stack traces (avoids e.g. sending
832
+ # a users home directory name if package is installed there)
833
+ for value in event.get("exception", {}).get("values", []):
834
+ for frame in value.get("stacktrace", {}).get("frames", []):
835
+ frame["abs_path"] = ""
836
+
837
+ if f"rasa_sdk{os.path.sep}executor.py" in frame["filename"]:
838
+ # this looks a lot like an exception in the SDK and hence custom code
839
+ # no need for us to deal with that
840
+ return None
841
+ elif "site-packages" in frame["filename"]:
842
+ # drop site-packages and following slash / backslash
843
+ relative_name = frame["filename"].split("site-packages")[-1][1:]
844
+ frame["filename"] = os.path.join("site-packages", relative_name)
845
+ elif "dist-packages" in frame["filename"]:
846
+ # drop dist-packages and following slash / backslash
847
+ relative_name = frame["filename"].split("dist-packages")[-1][1:]
848
+ frame["filename"] = os.path.join("dist-packages", relative_name)
849
+ elif os.path.isabs(frame["filename"]):
850
+ # if the file path is absolute, we'll drop the whole event as this is
851
+ # very likely custom code. needs to happen after cleaning as
852
+ # site-packages / dist-packages paths are also absolute, but fine.
853
+ return None
854
+ return event
855
+
856
+
857
+ @ensure_telemetry_enabled
858
+ def initialize_error_reporting() -> None:
859
+ """Sets up automated error reporting.
860
+
861
+ Exceptions are reported to sentry. We avoid sending any metadata (local
862
+ variables, paths, ...) to make sure we don't compromise any data. Only the
863
+ exception and its stacktrace is logged and only if the exception origins
864
+ from the `rasa` package.
865
+ """
866
+ import sentry_sdk
867
+ from sentry_sdk import configure_scope
868
+ from sentry_sdk.integrations.atexit import AtexitIntegration
869
+ from sentry_sdk.integrations.dedupe import DedupeIntegration
870
+ from sentry_sdk.integrations.excepthook import ExcepthookIntegration
871
+
872
+ # key for local testing can be found at
873
+ # https://sentry.io/settings/rasahq/projects/rasa-open-source/install/python/
874
+ # for local testing, set the key using `RASA_EXCEPTION_WRITE_KEY=key rasa <command>`
875
+ key = sentry_write_key()
876
+
877
+ if not key:
878
+ return
879
+
880
+ telemetry_id = get_telemetry_id()
881
+
882
+ # this is a very defensive configuration, avoiding as many integrations as
883
+ # possible. it also submits very little data (exception with error message
884
+ # and line numbers).
885
+ sentry_sdk.init(
886
+ f"https://{key}.ingest.sentry.io/2801673",
887
+ before_send=before_send,
888
+ integrations=[
889
+ ExcepthookIntegration(),
890
+ DedupeIntegration(),
891
+ AtexitIntegration(lambda _, __: None),
892
+ ],
893
+ send_default_pii=False, # activate PII filter
894
+ server_name=telemetry_id or "UNKNOWN",
895
+ ignore_errors=[
896
+ # std lib errors
897
+ KeyboardInterrupt, # user hit the interrupt key (Ctrl+C)
898
+ MemoryError, # machine is running out of memory
899
+ NotImplementedError, # user is using a feature that is not implemented
900
+ asyncio.CancelledError, # an async operation has been cancelled by the user
901
+ # expected Rasa errors
902
+ RasaException,
903
+ OSError,
904
+ ],
905
+ in_app_include=["rasa"], # only submit errors in this package
906
+ with_locals=False, # don't submit local variables
907
+ release=f"rasa-{rasa.__version__}",
908
+ default_integrations=False,
909
+ environment="development" if in_continuous_integration() else "production",
910
+ )
911
+
912
+ if not telemetry_id:
913
+ return
914
+
915
+ with configure_scope() as scope:
916
+ # sentry added these more recently, just a protection in a case where a
917
+ # user has installed an older version of sentry
918
+ if hasattr(scope, "set_user"):
919
+ scope.set_user({"id": telemetry_id})
920
+
921
+ default_context = _default_context_fields()
922
+ if hasattr(scope, "set_context"):
923
+ if "os" in default_context:
924
+ # os is a nested dict, hence we report it separately
925
+ scope.set_context("Operating System", default_context.pop("os"))
926
+ scope.set_context("Environment", default_context)
927
+
928
+
929
+ @contextlib.contextmanager
930
+ def track_model_training(
931
+ training_data: "TrainingDataImporter", model_type: Text, is_finetuning: bool = False
932
+ ) -> typing.Generator[None, None, None]:
933
+ """Track a model training started.
934
+
935
+ WARNING: since this is a generator, it can't use the ensure telemetry
936
+ decorator. We need to manually add these checks here. This can be
937
+ fixed as soon as we drop python 3.6 support.
938
+
939
+ Args:
940
+ training_data: Training data used for the training.
941
+ model_type: Specifies the type of training, should be either "rasa", "core"
942
+ or "nlu".
943
+ is_finetuning: `True` if the model is trained by finetuning another model.
944
+ """
945
+ if not initialize_telemetry():
946
+ # telemetry reporting is disabled. we won't do any reporting
947
+ yield # runs the training
948
+ return
949
+
950
+ config = training_data.get_config()
951
+ stories = training_data.get_stories()
952
+ nlu_data = training_data.get_nlu_data()
953
+ domain = training_data.get_domain()
954
+ flows = training_data.get_flows()
955
+ count_conditional_responses = domain.count_conditional_response_variations()
956
+ (
957
+ count_total_mappings,
958
+ count_custom_mappings,
959
+ count_conditional_mappings,
960
+ ) = domain.count_slot_mapping_statistics()
961
+
962
+ training_id = uuid.uuid4().hex
963
+
964
+ tracking_data = {
965
+ "language": config.get("language"),
966
+ "training_id": training_id,
967
+ "type": model_type,
968
+ "pipeline": config.get("pipeline"),
969
+ "policies": config.get("policies"),
970
+ "train_schema": config.get("train_schema"),
971
+ "predict_schema": config.get("predict_schema"),
972
+ "model_groups": rasa.core.utils.AvailableEndpoints.get_instance().model_groups,
973
+ "api_health_check_enabled": (
974
+ os.getenv(
975
+ LLM_API_HEALTH_CHECK_ENV_VAR, LLM_API_HEALTH_CHECK_DEFAULT_VALUE
976
+ ).lower()
977
+ == "true"
978
+ ),
979
+ "num_intent_examples": len(nlu_data.intent_examples),
980
+ "num_entity_examples": len(nlu_data.entity_examples),
981
+ "num_actions": len(domain.action_names_or_texts),
982
+ # Old nomenclature from when 'responses' were still called
983
+ # 'templates' in the domain
984
+ "num_templates": len(domain.responses),
985
+ "num_conditional_response_variations": count_conditional_responses,
986
+ "num_slot_mappings": count_total_mappings,
987
+ "num_custom_slot_mappings": count_custom_mappings,
988
+ "num_conditional_slot_mappings": count_conditional_mappings,
989
+ "num_slots": len(domain.slots),
990
+ "num_forms": len(domain.forms),
991
+ "num_intents": len(domain.intents),
992
+ "num_entities": len(domain.entities),
993
+ "num_story_steps": len(stories.story_steps),
994
+ "num_lookup_tables": len(nlu_data.lookup_tables),
995
+ "num_synonyms": len(nlu_data.entity_synonyms),
996
+ "num_regexes": len(nlu_data.regex_features),
997
+ "is_finetuning": is_finetuning,
998
+ "recipe": config.get("recipe"),
999
+ }
1000
+
1001
+ flow_statistics = _collect_flow_statistics(flows.underlying_flows)
1002
+ tracking_data.update(flow_statistics)
1003
+ command_generator_settings = _get_llm_command_generator_config(config)
1004
+ tracking_data.update(command_generator_settings)
1005
+
1006
+ # Make sure to update the example in docs/docs/telemetry/telemetry.mdx
1007
+ # if you change / add any properties
1008
+ _track(
1009
+ TRAINING_STARTED_EVENT,
1010
+ tracking_data,
1011
+ )
1012
+ start = datetime.now()
1013
+ yield
1014
+ runtime = datetime.now() - start
1015
+
1016
+ _track(
1017
+ TRAINING_COMPLETED_EVENT,
1018
+ {
1019
+ "training_id": training_id,
1020
+ "type": model_type,
1021
+ "runtime": int(runtime.total_seconds()),
1022
+ },
1023
+ )
1024
+
1025
+
1026
+ def _collect_flow_statistics(flows: List[Flow]) -> Dict[str, Any]:
1027
+ """Collects some statistics about the flows, such as number of specific steps."""
1028
+ data = {
1029
+ NUM_FLOWS: len(flows),
1030
+ NUM_FLOWS_WITH_NLU_TRIGGER: 0,
1031
+ NUM_FLOWS_WITH_FLOW_GUARDS: 0,
1032
+ NUM_FLOWS_ALWAYS_INCLUDED_IN_PROMPT: 0,
1033
+ NUM_FLOWS_WITH_NOT_STARTABLE_FLOW_GUARDS: 0,
1034
+ NUM_COLLECT_STEPS: 0,
1035
+ NUM_COLLECT_STEPS_WITH_SEPARATE_UTTER: 0,
1036
+ NUM_COLLECT_STEPS_WITH_REJECTIONS: 0,
1037
+ NUM_COLLECT_STEPS_WITH_NOT_RESET_AFTER_FLOW_ENDS: 0,
1038
+ NUM_SET_SLOT_STEPS: 0,
1039
+ MAX_DEPTH_OF_IF_CONSTRUCT: 0,
1040
+ NUM_LINK_STEPS: 0,
1041
+ NUM_CALL_STEPS: 0,
1042
+ NUM_SHARED_SLOTS_BETWEEN_FLOWS: 0,
1043
+ }
1044
+
1045
+ slots_used_in_different_flows = defaultdict(set)
1046
+
1047
+ for flow in flows:
1048
+ if flow.guard_condition:
1049
+ data[NUM_FLOWS_WITH_FLOW_GUARDS] += 1
1050
+ if flow.guard_condition.lower() == "false":
1051
+ data[NUM_FLOWS_WITH_NOT_STARTABLE_FLOW_GUARDS] += 1
1052
+
1053
+ if flow.always_include_in_prompt:
1054
+ data[NUM_FLOWS_ALWAYS_INCLUDED_IN_PROMPT] += 1
1055
+
1056
+ if flow.nlu_triggers:
1057
+ data[NUM_FLOWS_WITH_NLU_TRIGGER] += 1
1058
+
1059
+ for step in flow.steps_with_calls_resolved:
1060
+ if isinstance(step, CollectInformationFlowStep):
1061
+ slots_used_in_different_flows[step.collect].add(flow.id)
1062
+ data[NUM_COLLECT_STEPS] += 1
1063
+ if len(step.rejections) > 0:
1064
+ data[NUM_COLLECT_STEPS_WITH_REJECTIONS] += 1
1065
+ if not step.reset_after_flow_ends:
1066
+ data[NUM_COLLECT_STEPS_WITH_NOT_RESET_AFTER_FLOW_ENDS] += 1
1067
+ if step.utter != f"{UTTER_ASK_PREFIX}{step.collect}":
1068
+ data[NUM_COLLECT_STEPS_WITH_SEPARATE_UTTER] += 1
1069
+
1070
+ if isinstance(step, SetSlotsFlowStep):
1071
+ for slot in step.slots:
1072
+ slots_used_in_different_flows[slot["key"]].add(flow.id)
1073
+ data[NUM_SET_SLOT_STEPS] += 1
1074
+
1075
+ if isinstance(step, LinkFlowStep):
1076
+ data[NUM_LINK_STEPS] += 1
1077
+
1078
+ if isinstance(step, CallFlowStep):
1079
+ data[NUM_CALL_STEPS] += 1
1080
+
1081
+ if step.next:
1082
+ depth = step.next.depth_in_tree()
1083
+ if depth > data[MAX_DEPTH_OF_IF_CONSTRUCT]:
1084
+ data[MAX_DEPTH_OF_IF_CONSTRUCT] = depth
1085
+
1086
+ for flows_with_slot in slots_used_in_different_flows.values():
1087
+ if len(flows_with_slot) > 1:
1088
+ data[NUM_SHARED_SLOTS_BETWEEN_FLOWS] += 1
1089
+
1090
+ return data
1091
+
1092
+
1093
+ def _get_llm_command_generator_config(config: Dict[str, Any]) -> Optional[Dict]:
1094
+ """Returns the configuration for the LLMCommandGenerator.
1095
+
1096
+ Includes the model name, whether a custom prompt is used, whether flow
1097
+ retrieval is enabled, and flow retrieval embedding model.
1098
+ """
1099
+ from rasa.shared.constants import (
1100
+ EMBEDDINGS_CONFIG_KEY,
1101
+ MODEL_CONFIG_KEY,
1102
+ MODEL_NAME_CONFIG_KEY,
1103
+ )
1104
+ from rasa.dialogue_understanding.generator import (
1105
+ LLMCommandGenerator,
1106
+ SingleStepLLMCommandGenerator,
1107
+ MultiStepLLMCommandGenerator,
1108
+ )
1109
+ from rasa.dialogue_understanding.generator.multi_step.multi_step_llm_command_generator import ( # noqa: E501
1110
+ HANDLE_FLOWS_KEY,
1111
+ FILL_SLOTS_KEY,
1112
+ )
1113
+ from rasa.dialogue_understanding.generator.constants import (
1114
+ LLM_CONFIG_KEY,
1115
+ DEFAULT_LLM_CONFIG,
1116
+ FLOW_RETRIEVAL_KEY,
1117
+ )
1118
+ from rasa.dialogue_understanding.generator.flow_retrieval import (
1119
+ DEFAULT_EMBEDDINGS_CONFIG,
1120
+ )
1121
+
1122
+ def find_command_generator_component(pipeline: List) -> Optional[Dict]:
1123
+ """Finds the LLMCommandGenerator component in the pipeline."""
1124
+ for component in pipeline:
1125
+ if component["name"] in [
1126
+ LLMCommandGenerator.__name__,
1127
+ SingleStepLLMCommandGenerator.__name__,
1128
+ MultiStepLLMCommandGenerator.__name__,
1129
+ ]:
1130
+ return component
1131
+ return None
1132
+
1133
+ def extract_llm_command_generator_llm_client_settings(component: Dict) -> Dict:
1134
+ """Extracts settings related to LLM command generator."""
1135
+ llm_config = component.get(LLM_CONFIG_KEY, {})
1136
+ # Config at this stage is not yet resolved, so read from `model_group`
1137
+ llm_model_group_id = llm_config.get(MODEL_GROUP_CONFIG_KEY)
1138
+ llm_model_name = llm_config.get(MODEL_CONFIG_KEY) or llm_config.get(
1139
+ MODEL_NAME_CONFIG_KEY
1140
+ )
1141
+ if llm_model_group_id is None and llm_model_name is None:
1142
+ llm_model_name = DEFAULT_LLM_CONFIG[MODEL_CONFIG_KEY]
1143
+
1144
+ custom_prompt_used = (
1145
+ PROMPT_CONFIG_KEY in component or PROMPT_TEMPLATE_CONFIG_KEY in component
1146
+ )
1147
+ return {
1148
+ LLM_COMMAND_GENERATOR_MODEL_NAME: llm_model_name,
1149
+ LLM_COMMAND_GENERATOR_MODEL_GROUP_ID: llm_model_group_id,
1150
+ LLM_COMMAND_GENERATOR_CUSTOM_PROMPT_USED: custom_prompt_used,
1151
+ }
1152
+
1153
+ def extract_multistep_command_generator_prompt_settings(component: Dict) -> Dict:
1154
+ """Extracts settings related to multistep command generator."""
1155
+ prompt_templates = component.get("prompt_templates", {})
1156
+ handle_flows_prompt_used = HANDLE_FLOWS_KEY in prompt_templates
1157
+ fill_slots_prompt_used = FILL_SLOTS_KEY in prompt_templates
1158
+ return {
1159
+ MULTI_STEP_LLM_COMMAND_GENERATOR_HANDLE_FLOWS_PROMPT_USED: handle_flows_prompt_used, # noqa: E501
1160
+ MULTI_STEP_LLM_COMMAND_GENERATOR_FILL_SLOTS_PROMPT_USED: fill_slots_prompt_used, # noqa: E501
1161
+ }
1162
+
1163
+ def extract_flow_retrieval_settings(component: Dict) -> Dict:
1164
+ """Extracts settings related to flow retrieval."""
1165
+ flow_retrieval_config = component.get(FLOW_RETRIEVAL_KEY, {})
1166
+ flow_retrieval_enabled = flow_retrieval_config.get("active", True)
1167
+ embeddings_config = flow_retrieval_config.get(
1168
+ EMBEDDINGS_CONFIG_KEY, DEFAULT_EMBEDDINGS_CONFIG
1169
+ )
1170
+ flow_retrieval_embedding_model_name = (
1171
+ (
1172
+ embeddings_config.get(MODEL_NAME_CONFIG_KEY)
1173
+ or embeddings_config.get(MODEL_CONFIG_KEY)
1174
+ )
1175
+ if flow_retrieval_enabled
1176
+ else None
1177
+ )
1178
+ # Config at this stage is not yet resolved, so read from `model_group`
1179
+ flow_retrieval_embedding_model_group_id = embeddings_config.get(
1180
+ MODEL_GROUP_CONFIG_KEY
1181
+ )
1182
+ return {
1183
+ FLOW_RETRIEVAL_ENABLED: flow_retrieval_enabled,
1184
+ FLOW_RETRIEVAL_EMBEDDING_MODEL_NAME: flow_retrieval_embedding_model_name,
1185
+ FLOW_RETRIEVAL_EMBEDDING_MODEL_GROUP_ID: flow_retrieval_embedding_model_group_id, # noqa: E501
1186
+ }
1187
+
1188
+ def extract_settings(component: Dict) -> Dict:
1189
+ """Extracts the settings from the command generator component."""
1190
+ settings = {}
1191
+ settings.update(extract_llm_command_generator_llm_client_settings(component))
1192
+ settings.update(extract_multistep_command_generator_prompt_settings(component))
1193
+ settings.update(extract_flow_retrieval_settings(component))
1194
+ return settings
1195
+
1196
+ command_generator_config = {
1197
+ LLM_COMMAND_GENERATOR_MODEL_NAME: None,
1198
+ LLM_COMMAND_GENERATOR_MODEL_GROUP_ID: None,
1199
+ LLM_COMMAND_GENERATOR_CUSTOM_PROMPT_USED: None,
1200
+ MULTI_STEP_LLM_COMMAND_GENERATOR_HANDLE_FLOWS_PROMPT_USED: None,
1201
+ MULTI_STEP_LLM_COMMAND_GENERATOR_FILL_SLOTS_PROMPT_USED: None,
1202
+ FLOW_RETRIEVAL_ENABLED: None,
1203
+ FLOW_RETRIEVAL_EMBEDDING_MODEL_NAME: None,
1204
+ FLOW_RETRIEVAL_EMBEDDING_MODEL_GROUP_ID: None,
1205
+ }
1206
+
1207
+ pipeline = config.get("pipeline", [])
1208
+ if not isinstance(pipeline, list):
1209
+ return command_generator_config
1210
+
1211
+ command_generator_component = find_command_generator_component(pipeline)
1212
+ if command_generator_component is not None:
1213
+ extracted_settings = extract_settings(command_generator_component)
1214
+ command_generator_config.update(extracted_settings)
1215
+
1216
+ return command_generator_config
1217
+
1218
+
1219
+ @ensure_telemetry_enabled
1220
+ def track_telemetry_disabled() -> None:
1221
+ """Track when a user disables telemetry."""
1222
+ _track(TELEMETRY_DISABLED_EVENT)
1223
+
1224
+
1225
+ @ensure_telemetry_enabled
1226
+ def track_data_split(fraction: float, data_type: Text) -> None:
1227
+ """Track when a user splits data.
1228
+
1229
+ Args:
1230
+ fraction: How much data goes into train and how much goes into test
1231
+ data_type: Is this core, nlu or nlg data
1232
+ """
1233
+ _track(TELEMETRY_DATA_SPLIT_EVENT, {"fraction": fraction, "type": data_type})
1234
+
1235
+
1236
+ @ensure_telemetry_enabled
1237
+ def track_validate_files(validation_success: bool) -> None:
1238
+ """Track when a user validates data files.
1239
+
1240
+ Args:
1241
+ validation_success: Whether the validation was successful
1242
+ """
1243
+ _track(TELEMETRY_DATA_VALIDATED_EVENT, {"validation_success": validation_success})
1244
+
1245
+
1246
+ @ensure_telemetry_enabled
1247
+ def track_data_convert(output_format: Text, data_type: Text) -> None:
1248
+ """Track when a user converts data.
1249
+
1250
+ Args:
1251
+ output_format: Target format for the converter
1252
+ data_type: Is this core, nlu or nlg data
1253
+ """
1254
+ _track(
1255
+ TELEMETRY_DATA_CONVERTED_EVENT,
1256
+ {"output_format": output_format, "type": data_type},
1257
+ )
1258
+
1259
+
1260
+ @ensure_telemetry_enabled
1261
+ def track_tracker_export(
1262
+ number_of_exported_events: int,
1263
+ tracker_store: "TrackerStore",
1264
+ event_broker: "EventBroker",
1265
+ ) -> None:
1266
+ """Track when a user exports trackers.
1267
+
1268
+ Args:
1269
+ number_of_exported_events: Number of events that got exported
1270
+ tracker_store: Store used to retrieve the events from
1271
+ event_broker: Broker the events are getting published towards
1272
+ """
1273
+ _track(
1274
+ TELEMETRY_TRACKER_EXPORTED_EVENT,
1275
+ {
1276
+ "number_of_exported_events": number_of_exported_events,
1277
+ "tracker_store": type(tracker_store).__name__,
1278
+ "event_broker": type(event_broker).__name__,
1279
+ },
1280
+ )
1281
+
1282
+
1283
+ @ensure_telemetry_enabled
1284
+ def track_interactive_learning_start(
1285
+ skip_visualization: bool, save_in_e2e: bool
1286
+ ) -> None:
1287
+ """Track when a user starts an interactive learning session.
1288
+
1289
+ Args:
1290
+ skip_visualization: Is visualization skipped in this session
1291
+ save_in_e2e: Is e2e used in this session
1292
+ """
1293
+ _track(
1294
+ TELEMETRY_INTERACTIVE_LEARNING_STARTED_EVENT,
1295
+ {"skip_visualization": skip_visualization, "save_in_e2e": save_in_e2e},
1296
+ )
1297
+
1298
+
1299
+ @ensure_telemetry_enabled
1300
+ def track_server_start(
1301
+ input_channels: List["InputChannel"],
1302
+ endpoints: Optional["AvailableEndpoints"],
1303
+ model_directory: Optional[Text],
1304
+ number_of_workers: int,
1305
+ is_api_enabled: bool,
1306
+ ) -> None:
1307
+ """Tracks when a user starts a rasa server.
1308
+
1309
+ Args:
1310
+ input_channels: Used input channels
1311
+ endpoints: Endpoint configuration for the server
1312
+ model_directory: directory of the running model
1313
+ number_of_workers: number of used Sanic workers
1314
+ is_api_enabled: whether the rasa API server is enabled
1315
+ """
1316
+ from rasa.core.utils import AvailableEndpoints
1317
+
1318
+ def project_fingerprint_from_model(
1319
+ _model_directory: Optional[Text],
1320
+ ) -> Optional[Text]:
1321
+ """Gets project fingerprint from an app's loaded model."""
1322
+ if not model_directory:
1323
+ return None
1324
+
1325
+ try:
1326
+ model_archive = model.get_local_model(_model_directory)
1327
+ metadata = LocalModelStorage.metadata_from_archive(model_archive)
1328
+
1329
+ return metadata.project_fingerprint
1330
+ except Exception:
1331
+ return None
1332
+
1333
+ if not endpoints:
1334
+ endpoints = AvailableEndpoints()
1335
+
1336
+ _track(
1337
+ TELEMETRY_SERVER_STARTED_EVENT,
1338
+ {
1339
+ "input_channels": [i.name() for i in input_channels],
1340
+ "api_enabled": is_api_enabled,
1341
+ "number_of_workers": number_of_workers,
1342
+ "endpoints_nlg": endpoints.nlg.type if endpoints.nlg else None,
1343
+ "endpoints_nlu": endpoints.nlu.type if endpoints.nlu else None,
1344
+ "endpoints_action_server": endpoints.action.type
1345
+ if endpoints.action
1346
+ else None,
1347
+ "endpoints_model_server": endpoints.model.type if endpoints.model else None,
1348
+ "endpoints_tracker_store": endpoints.tracker_store.type
1349
+ if endpoints.tracker_store
1350
+ else None,
1351
+ "endpoints_lock_store": endpoints.lock_store.type
1352
+ if endpoints.lock_store
1353
+ else None,
1354
+ "endpoints_event_broker": endpoints.event_broker.type
1355
+ if endpoints.event_broker
1356
+ else None,
1357
+ "project": project_fingerprint_from_model(model_directory),
1358
+ },
1359
+ )
1360
+
1361
+
1362
+ @ensure_telemetry_enabled
1363
+ def track_project_init(path: Text) -> None:
1364
+ """Track when a user creates a project using rasa init.
1365
+
1366
+ Args:
1367
+ path: Location of the project
1368
+ """
1369
+ _track(
1370
+ TELEMETRY_PROJECT_CREATED_EVENT, {"init_directory": _hash_directory_path(path)}
1371
+ )
1372
+
1373
+
1374
+ @ensure_telemetry_enabled
1375
+ def track_shell_started(model_type: Text) -> None:
1376
+ """Track when a user starts a bot using rasa shell.
1377
+
1378
+ Args:
1379
+ model_type: Type of the model, core / nlu or rasa.
1380
+ """
1381
+ _track(TELEMETRY_SHELL_STARTED_EVENT, {"type": model_type})
1382
+
1383
+
1384
+ @ensure_telemetry_enabled
1385
+ def track_inspect_started(model_type: Text) -> None:
1386
+ """Track when a user starts a bot using rasa inspect.
1387
+
1388
+ Args:
1389
+ channel: Channel name `socketio` (used for chat assistants)
1390
+ or `browser_audio` (used for voice).
1391
+ """
1392
+ _track(TELEMETRY_INSPECT_STARTED_EVENT, {"type": model_type})
1393
+
1394
+
1395
+ @ensure_telemetry_enabled
1396
+ def track_visualization() -> None:
1397
+ """Track when a user runs the visualization."""
1398
+ _track(TELEMETRY_VISUALIZATION_STARTED_EVENT)
1399
+
1400
+
1401
+ @ensure_telemetry_enabled
1402
+ def track_core_model_test(num_story_steps: int, e2e: bool, agent: "Agent") -> None:
1403
+ """Track when a user tests a core model.
1404
+
1405
+ Args:
1406
+ num_story_steps: Number of test stories used for the comparison
1407
+ e2e: indicator if tests running in end to end mode
1408
+ agent: Agent of the model getting tested
1409
+ """
1410
+ if agent.processor is None:
1411
+ project_fingerprint = ""
1412
+ else:
1413
+ project_fingerprint = agent.processor.model_metadata.project_fingerprint
1414
+
1415
+ _track(
1416
+ TELEMETRY_TEST_CORE_EVENT,
1417
+ {
1418
+ "project": project_fingerprint,
1419
+ "end_to_end": e2e,
1420
+ "num_story_steps": num_story_steps,
1421
+ },
1422
+ )
1423
+
1424
+
1425
+ @ensure_telemetry_enabled
1426
+ def track_nlu_model_test(test_data: "TrainingData") -> None:
1427
+ """Track when a user tests an nlu model.
1428
+
1429
+ Args:
1430
+ test_data: Data used for testing
1431
+ """
1432
+ _track(
1433
+ TELEMETRY_TEST_NLU_EVENT,
1434
+ {
1435
+ "num_intent_examples": len(test_data.intent_examples),
1436
+ "num_entity_examples": len(test_data.entity_examples),
1437
+ "num_lookup_tables": len(test_data.lookup_tables),
1438
+ "num_synonyms": len(test_data.entity_synonyms),
1439
+ "num_regexes": len(test_data.regex_features),
1440
+ },
1441
+ )
1442
+
1443
+
1444
+ @ensure_telemetry_enabled
1445
+ def track_markers_extraction_initiated(
1446
+ strategy: Text, only_extract: bool, seed: bool, count: Optional[int]
1447
+ ) -> None:
1448
+ """Track when a user tries to extract success markers.
1449
+
1450
+ Args:
1451
+ strategy: The strategy the user is using for tracker selection
1452
+ only_extract: Indicates if the user is only extracting markers or also
1453
+ producing stats
1454
+ seed: Indicates if the user used a seed for this attempt
1455
+ count: (Optional) The number of trackers the user is trying to select.
1456
+ """
1457
+ _track(
1458
+ TELEMETRY_MARKERS_EXTRACTION_INITIATED_EVENT,
1459
+ {
1460
+ "strategy": strategy,
1461
+ "only_extract": only_extract,
1462
+ "seed": seed,
1463
+ "count": count,
1464
+ },
1465
+ )
1466
+
1467
+
1468
+ @ensure_telemetry_enabled
1469
+ def track_markers_extracted(trackers_count: int) -> None:
1470
+ """Track when markers have been extracted by a user.
1471
+
1472
+ Args:
1473
+ trackers_count: The actual number of trackers processed
1474
+ """
1475
+ _track(TELEMETRY_MARKERS_EXTRACTED_EVENT, {"trackers_count": trackers_count})
1476
+
1477
+
1478
+ @ensure_telemetry_enabled
1479
+ def track_markers_stats_computed(trackers_count: int) -> None:
1480
+ """Track when stats over markers have been computed by a user.
1481
+
1482
+ Args:
1483
+ trackers_count: The actual number of trackers processed
1484
+ """
1485
+ _track(TELEMETRY_MARKERS_STATS_COMPUTED_EVENT, {"trackers_count": trackers_count})
1486
+
1487
+
1488
+ @ensure_telemetry_enabled
1489
+ def track_markers_parsed_count(
1490
+ marker_count: int, max_depth: int, branching_factor: int
1491
+ ) -> None:
1492
+ """Track when markers have been successfully parsed from config.
1493
+
1494
+ Args:
1495
+ marker_count: The number of markers found in the config
1496
+ max_depth: The maximum depth of any marker in the config
1497
+ branching_factor: The maximum number of children of any marker in the config.
1498
+ """
1499
+ _track(
1500
+ TELEMETRY_MARKERS_PARSED_COUNT,
1501
+ {
1502
+ "marker_count": marker_count,
1503
+ "max_depth": max_depth,
1504
+ "branching_factor": branching_factor,
1505
+ },
1506
+ )
1507
+
1508
+
1509
+ def extract_assertion_type_counts(
1510
+ input_test_cases: List["TestCase"],
1511
+ ) -> typing.Tuple[bool, Dict[str, Any]]:
1512
+ """Extracts the total count of different assertion types from the test cases."""
1513
+ from rasa.e2e_test.assertions import AssertionType
1514
+
1515
+ uses_assertions = False
1516
+
1517
+ flow_started_count = 0
1518
+ flow_completed_count = 0
1519
+ flow_cancelled_count = 0
1520
+ pattern_clarification_contains_count = 0
1521
+ action_executed_count = 0
1522
+ slot_was_set_count = 0
1523
+ slot_was_not_set_count = 0
1524
+ bot_uttered_count = 0
1525
+ generative_response_is_relevant_count = 0
1526
+ generative_response_is_grounded_count = 0
1527
+
1528
+ for test_case in input_test_cases:
1529
+ for step in test_case.steps:
1530
+ assertions = step.assertions if step.assertions else []
1531
+ for assertion in assertions:
1532
+ if assertion.type == AssertionType.ACTION_EXECUTED.value:
1533
+ action_executed_count += 1
1534
+ elif assertion.type == AssertionType.SLOT_WAS_SET.value:
1535
+ slot_was_set_count += 1
1536
+ elif assertion.type == AssertionType.SLOT_WAS_NOT_SET.value:
1537
+ slot_was_not_set_count += 1
1538
+ elif assertion.type == AssertionType.BOT_UTTERED.value:
1539
+ bot_uttered_count += 1
1540
+ elif (
1541
+ assertion.type
1542
+ == AssertionType.GENERATIVE_RESPONSE_IS_RELEVANT.value
1543
+ ):
1544
+ generative_response_is_relevant_count += 1
1545
+ elif (
1546
+ assertion.type
1547
+ == AssertionType.GENERATIVE_RESPONSE_IS_GROUNDED.value
1548
+ ):
1549
+ generative_response_is_grounded_count += 1
1550
+ elif assertion.type == AssertionType.FLOW_STARTED.value:
1551
+ flow_started_count += 1
1552
+ elif assertion.type == AssertionType.FLOW_COMPLETED.value:
1553
+ flow_completed_count += 1
1554
+ elif assertion.type == AssertionType.FLOW_CANCELLED.value:
1555
+ flow_cancelled_count += 1
1556
+ elif (
1557
+ assertion.type == AssertionType.PATTERN_CLARIFICATION_CONTAINS.value
1558
+ ):
1559
+ pattern_clarification_contains_count += 1
1560
+
1561
+ uses_assertions = True
1562
+
1563
+ result = {
1564
+ "flow_started_count": flow_started_count,
1565
+ "flow_completed_count": flow_completed_count,
1566
+ "flow_cancelled_count": flow_cancelled_count,
1567
+ "pattern_clarification_contains_count": pattern_clarification_contains_count,
1568
+ "action_executed_count": action_executed_count,
1569
+ "slot_was_set_count": slot_was_set_count,
1570
+ "slot_was_not_set_count": slot_was_not_set_count,
1571
+ "bot_uttered_count": bot_uttered_count,
1572
+ "generative_response_is_relevant_count": generative_response_is_relevant_count,
1573
+ "generative_response_is_grounded_count": generative_response_is_grounded_count,
1574
+ }
1575
+
1576
+ return uses_assertions, result
1577
+
1578
+
1579
+ @ensure_telemetry_enabled
1580
+ def track_e2e_test_run(
1581
+ input_test_cases: List["TestCase"],
1582
+ input_fixtures: List["Fixture"],
1583
+ input_metadata: List["Metadata"],
1584
+ ) -> None:
1585
+ """Track an end-to-end test run."""
1586
+ properties = {
1587
+ "number_of_test_cases": len(input_test_cases),
1588
+ "number_of_fixtures": len(input_fixtures),
1589
+ "uses_fixtures": len(input_fixtures) > 0,
1590
+ "uses_metadata": len(input_metadata) > 0,
1591
+ "number_of_metadata": len(input_metadata),
1592
+ }
1593
+
1594
+ uses_assertions, assertion_type_counts = extract_assertion_type_counts(
1595
+ input_test_cases
1596
+ )
1597
+
1598
+ properties.update({"uses_assertions": uses_assertions})
1599
+
1600
+ if uses_assertions:
1601
+ properties.update(assertion_type_counts)
1602
+
1603
+ _track(
1604
+ TELEMETRY_E2E_TEST_RUN_STARTED_EVENT,
1605
+ properties,
1606
+ )
1607
+
1608
+
1609
+ @ensure_telemetry_enabled
1610
+ def track_response_rephrase(
1611
+ rephrase_all: bool,
1612
+ custom_prompt_template: Optional[str],
1613
+ llm_type: Optional[str],
1614
+ llm_model: Optional[str],
1615
+ llm_model_group_id: Optional[str],
1616
+ ) -> None:
1617
+ """Track when a user rephrases a response."""
1618
+ _track(
1619
+ TELEMETRY_RESPONSE_REPHRASED_EVENT,
1620
+ {
1621
+ "rephrase_all": rephrase_all,
1622
+ "custom_prompt_template": custom_prompt_template,
1623
+ "llm_type": llm_type,
1624
+ "llm_model": llm_model,
1625
+ "llm_model_group_id": llm_model_group_id,
1626
+ },
1627
+ )
1628
+
1629
+
1630
+ @ensure_telemetry_enabled
1631
+ def track_intentless_policy_train() -> None:
1632
+ """Track when a user trains a policy."""
1633
+ _track(TELEMETRY_INTENTLESS_POLICY_TRAINING_STARTED_EVENT)
1634
+
1635
+
1636
+ @ensure_telemetry_enabled
1637
+ def track_intentless_policy_train_completed(
1638
+ embeddings_type: Optional[str],
1639
+ embeddings_model: Optional[str],
1640
+ embeddings_model_group_id: Optional[str],
1641
+ llm_type: Optional[str],
1642
+ llm_model: Optional[str],
1643
+ llm_model_group_id: Optional[str],
1644
+ ) -> None:
1645
+ """Track when a user trains a policy."""
1646
+ _track(
1647
+ TELEMETRY_INTENTLESS_POLICY_TRAINING_COMPLETED_EVENT,
1648
+ {
1649
+ "embeddings_type": embeddings_type,
1650
+ "embeddings_model": embeddings_model,
1651
+ "embeddings_model_group_id": embeddings_model_group_id,
1652
+ "llm_type": llm_type,
1653
+ "llm_model": llm_model,
1654
+ "llm_model_group_id": llm_model_group_id,
1655
+ },
1656
+ )
1657
+
1658
+
1659
+ @ensure_telemetry_enabled
1660
+ def track_intentless_policy_predict(
1661
+ embeddings_type: Optional[str],
1662
+ embeddings_model: Optional[str],
1663
+ embeddings_model_group_id: Optional[str],
1664
+ llm_type: Optional[str],
1665
+ llm_model: Optional[str],
1666
+ llm_model_group_id: Optional[str],
1667
+ score: float,
1668
+ ) -> None:
1669
+ """Track when a user trains a policy."""
1670
+ _track(
1671
+ TELEMETRY_INTENTLESS_POLICY_PREDICT_EVENT,
1672
+ {
1673
+ "embeddings_type": embeddings_type,
1674
+ "embeddings_model": embeddings_model,
1675
+ "embeddings_model_group_id": embeddings_model_group_id,
1676
+ "llm_type": llm_type,
1677
+ "llm_model": llm_model,
1678
+ "llm_model_group_id": llm_model_group_id,
1679
+ "score": score,
1680
+ },
1681
+ )
1682
+
1683
+
1684
+ @ensure_telemetry_enabled
1685
+ def identify_endpoint_config_traits(
1686
+ endpoints_file: Optional[Text],
1687
+ context: Optional[Dict[Text, Any]] = None,
1688
+ ) -> None:
1689
+ """Collect traits if enabled.
1690
+
1691
+ Otherwise, sets traits to None.
1692
+ """
1693
+ traits: Dict[str, Any] = {}
1694
+
1695
+ traits = append_tracing_trait(traits, endpoints_file)
1696
+ traits = append_metrics_trait(traits, endpoints_file)
1697
+ traits = append_anonymization_trait(traits, endpoints_file)
1698
+
1699
+ _identify(traits, context)
1700
+
1701
+
1702
+ def append_tracing_trait(
1703
+ traits: Dict[str, Any], endpoints_file: Optional[str]
1704
+ ) -> Dict[str, Any]:
1705
+ """Append the tracing trait to the traits dictionary."""
1706
+ import rasa.utils.endpoints
1707
+ from rasa.tracing.constants import ENDPOINTS_TRACING_KEY
1708
+
1709
+ tracing_config = rasa.utils.endpoints.read_endpoint_config(
1710
+ endpoints_file, ENDPOINTS_TRACING_KEY
1711
+ )
1712
+ traits[TRACING_BACKEND] = (
1713
+ tracing_config.type if tracing_config is not None else None
1714
+ )
1715
+
1716
+ return traits
1717
+
1718
+
1719
+ def append_metrics_trait(
1720
+ traits: Dict[str, Any], endpoints_file: Optional[str]
1721
+ ) -> Dict[str, Any]:
1722
+ """Append the metrics trait to the traits dictionary."""
1723
+ import rasa.utils.endpoints
1724
+ from rasa.tracing.constants import ENDPOINTS_METRICS_KEY
1725
+
1726
+ metrics_config = rasa.utils.endpoints.read_endpoint_config(
1727
+ endpoints_file, ENDPOINTS_METRICS_KEY
1728
+ )
1729
+ traits[METRICS_BACKEND] = (
1730
+ metrics_config.type if metrics_config is not None else None
1731
+ )
1732
+
1733
+ return traits
1734
+
1735
+
1736
+ def append_anonymization_trait(
1737
+ traits: Dict[str, Any], endpoints_file: Optional[str]
1738
+ ) -> Dict[str, Any]:
1739
+ """Append the anonymization trait to the traits dictionary."""
1740
+ from rasa.anonymization.anonymisation_rule_yaml_reader import (
1741
+ KEY_ANONYMIZATION_RULES,
1742
+ )
1743
+
1744
+ anonymization_config = rasa.anonymization.utils.read_endpoint_config(
1745
+ endpoints_file, KEY_ANONYMIZATION_RULES
1746
+ )
1747
+
1748
+ traits[KEY_ANONYMIZATION_RULES] = (
1749
+ rasa.anonymization.utils.extract_anonymization_traits(
1750
+ anonymization_config, KEY_ANONYMIZATION_RULES
1751
+ )
1752
+ )
1753
+
1754
+ return traits
1755
+
1756
+
1757
+ @ensure_telemetry_enabled
1758
+ def track_enterprise_search_policy_train_started() -> None:
1759
+ """Track when a user starts training Enterprise Search policy."""
1760
+ _track(TELEMETRY_ENTERPRISE_SEARCH_POLICY_TRAINING_STARTED_EVENT)
1761
+
1762
+
1763
+ @ensure_telemetry_enabled
1764
+ def track_enterprise_search_policy_train_completed(
1765
+ vector_store_type: Optional[str],
1766
+ embeddings_type: Optional[str],
1767
+ embeddings_model: Optional[str],
1768
+ embeddings_model_group_id: Optional[str],
1769
+ llm_type: Optional[str],
1770
+ llm_model: Optional[str],
1771
+ llm_model_group_id: Optional[str],
1772
+ citation_enabled: Optional[bool],
1773
+ ) -> None:
1774
+ """Track when a user completes training Enterprise Search policy."""
1775
+ _track(
1776
+ TELEMETRY_ENTERPRISE_SEARCH_POLICY_TRAINING_COMPLETED_EVENT,
1777
+ {
1778
+ "vector_store_type": vector_store_type,
1779
+ "embeddings_type": embeddings_type,
1780
+ "embeddings_model": embeddings_model,
1781
+ "embeddings_model_group_id": embeddings_model_group_id,
1782
+ "llm_type": llm_type,
1783
+ "llm_model": llm_model,
1784
+ "llm_model_group_id": llm_model_group_id,
1785
+ "citation_enabled": citation_enabled,
1786
+ },
1787
+ )
1788
+
1789
+
1790
+ @ensure_telemetry_enabled
1791
+ def track_enterprise_search_policy_predict(
1792
+ vector_store_type: Optional[str],
1793
+ embeddings_type: Optional[str],
1794
+ embeddings_model: Optional[str],
1795
+ embeddings_model_group_id: Optional[str],
1796
+ llm_type: Optional[str],
1797
+ llm_model: Optional[str],
1798
+ llm_model_group_id: Optional[str],
1799
+ citation_enabled: Optional[bool],
1800
+ ) -> None:
1801
+ """Track when a user predicts the next action using Enterprise Search policy."""
1802
+ _track(
1803
+ TELEMETRY_ENTERPRISE_SEARCH_POLICY_PREDICT_EVENT,
1804
+ {
1805
+ "vector_store_type": vector_store_type,
1806
+ "embeddings_type": embeddings_type,
1807
+ "embeddings_model": embeddings_model,
1808
+ "embeddings_model_group_id": embeddings_model_group_id,
1809
+ "llm_type": llm_type,
1810
+ "llm_model": llm_model,
1811
+ "llm_model_group_id": llm_model_group_id,
1812
+ "citation_enabled": citation_enabled,
1813
+ },
1814
+ )
1815
+
1816
+
1817
+ @ensure_telemetry_enabled
1818
+ def track_conversation_count_hard_limit(
1819
+ conversation_count: int, tracked_month: datetime
1820
+ ) -> None:
1821
+ """Track when the number of conversations reaches the hard limit."""
1822
+ _track(
1823
+ TELEMETRY_CONVERSATION_HARD_LIMIT_REACHED,
1824
+ {
1825
+ "conversation_count": conversation_count,
1826
+ "year": tracked_month.year,
1827
+ "month": tracked_month.month,
1828
+ },
1829
+ )
1830
+
1831
+
1832
+ @ensure_telemetry_enabled
1833
+ def track_conversation_count_soft_limit(
1834
+ conversation_count: int, tracked_month: datetime
1835
+ ) -> None:
1836
+ """Track when the number of conversations reaches the soft limit."""
1837
+ _track(
1838
+ TELEMETRY_CONVERSATION_SOFT_LIMIT_REACHED,
1839
+ {
1840
+ "conversation_count": conversation_count,
1841
+ "year": tracked_month.year,
1842
+ "month": tracked_month.month,
1843
+ },
1844
+ )
1845
+
1846
+
1847
+ @ensure_telemetry_enabled
1848
+ def track_conversation_count(conversation_count: int, tracked_month: datetime) -> None:
1849
+ """Track the number of conversations."""
1850
+ _track(
1851
+ TELEMETRY_CONVERSATION_COUNT,
1852
+ {
1853
+ "conversation_count": conversation_count,
1854
+ "year": tracked_month.year,
1855
+ "month": tracked_month.month,
1856
+ },
1857
+ )
1858
+
1859
+
1860
+ @ensure_telemetry_enabled
1861
+ def track_e2e_test_conversion_completed(file_type: str, test_case_count: int) -> None:
1862
+ """Track the used input file type for E2E test conversion."""
1863
+ _track(
1864
+ TELEMETRY_E2E_TEST_CONVERSION_EVENT,
1865
+ {
1866
+ E2E_TEST_CONVERSION_FILE_TYPE: file_type,
1867
+ E2E_TEST_CONVERSION_TEST_CASE_COUNT: test_case_count,
1868
+ },
1869
+ )