rasa-pro 3.9.18__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 (662) hide show
  1. README.md +415 -0
  2. rasa/__init__.py +10 -0
  3. rasa/__main__.py +156 -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 +146 -0
  12. rasa/cli/__init__.py +5 -0
  13. rasa/cli/arguments/__init__.py +0 -0
  14. rasa/cli/arguments/data.py +81 -0
  15. rasa/cli/arguments/default_arguments.py +165 -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 +204 -0
  20. rasa/cli/arguments/shell.py +13 -0
  21. rasa/cli/arguments/test.py +211 -0
  22. rasa/cli/arguments/train.py +263 -0
  23. rasa/cli/arguments/visualize.py +34 -0
  24. rasa/cli/arguments/x.py +30 -0
  25. rasa/cli/data.py +292 -0
  26. rasa/cli/e2e_test.py +586 -0
  27. rasa/cli/evaluate.py +222 -0
  28. rasa/cli/export.py +250 -0
  29. rasa/cli/inspect.py +63 -0
  30. rasa/cli/interactive.py +164 -0
  31. rasa/cli/license.py +65 -0
  32. rasa/cli/markers.py +78 -0
  33. rasa/cli/project_templates/__init__.py +0 -0
  34. rasa/cli/project_templates/calm/actions/__init__.py +0 -0
  35. rasa/cli/project_templates/calm/actions/action_template.py +27 -0
  36. rasa/cli/project_templates/calm/actions/add_contact.py +30 -0
  37. rasa/cli/project_templates/calm/actions/db.py +57 -0
  38. rasa/cli/project_templates/calm/actions/list_contacts.py +22 -0
  39. rasa/cli/project_templates/calm/actions/remove_contact.py +35 -0
  40. rasa/cli/project_templates/calm/config.yml +12 -0
  41. rasa/cli/project_templates/calm/credentials.yml +33 -0
  42. rasa/cli/project_templates/calm/data/flows/add_contact.yml +31 -0
  43. rasa/cli/project_templates/calm/data/flows/list_contacts.yml +14 -0
  44. rasa/cli/project_templates/calm/data/flows/remove_contact.yml +29 -0
  45. rasa/cli/project_templates/calm/db/contacts.json +10 -0
  46. rasa/cli/project_templates/calm/domain/add_contact.yml +39 -0
  47. rasa/cli/project_templates/calm/domain/list_contacts.yml +17 -0
  48. rasa/cli/project_templates/calm/domain/remove_contact.yml +38 -0
  49. rasa/cli/project_templates/calm/domain/shared.yml +10 -0
  50. rasa/cli/project_templates/calm/e2e_tests/cancelations/user_cancels_during_a_correction.yml +16 -0
  51. rasa/cli/project_templates/calm/e2e_tests/cancelations/user_changes_mind_on_a_whim.yml +7 -0
  52. rasa/cli/project_templates/calm/e2e_tests/corrections/user_corrects_contact_handle.yml +20 -0
  53. rasa/cli/project_templates/calm/e2e_tests/corrections/user_corrects_contact_name.yml +19 -0
  54. rasa/cli/project_templates/calm/e2e_tests/happy_paths/user_adds_contact_to_their_list.yml +15 -0
  55. rasa/cli/project_templates/calm/e2e_tests/happy_paths/user_lists_contacts.yml +5 -0
  56. rasa/cli/project_templates/calm/e2e_tests/happy_paths/user_removes_contact.yml +11 -0
  57. rasa/cli/project_templates/calm/e2e_tests/happy_paths/user_removes_contact_from_list.yml +12 -0
  58. rasa/cli/project_templates/calm/endpoints.yml +45 -0
  59. rasa/cli/project_templates/default/actions/__init__.py +0 -0
  60. rasa/cli/project_templates/default/actions/actions.py +27 -0
  61. rasa/cli/project_templates/default/config.yml +44 -0
  62. rasa/cli/project_templates/default/credentials.yml +33 -0
  63. rasa/cli/project_templates/default/data/nlu.yml +91 -0
  64. rasa/cli/project_templates/default/data/rules.yml +13 -0
  65. rasa/cli/project_templates/default/data/stories.yml +30 -0
  66. rasa/cli/project_templates/default/domain.yml +34 -0
  67. rasa/cli/project_templates/default/endpoints.yml +42 -0
  68. rasa/cli/project_templates/default/tests/test_stories.yml +91 -0
  69. rasa/cli/project_templates/tutorial/actions.py +22 -0
  70. rasa/cli/project_templates/tutorial/config.yml +11 -0
  71. rasa/cli/project_templates/tutorial/credentials.yml +33 -0
  72. rasa/cli/project_templates/tutorial/data/flows.yml +8 -0
  73. rasa/cli/project_templates/tutorial/data/patterns.yml +6 -0
  74. rasa/cli/project_templates/tutorial/domain.yml +21 -0
  75. rasa/cli/project_templates/tutorial/endpoints.yml +45 -0
  76. rasa/cli/run.py +135 -0
  77. rasa/cli/scaffold.py +269 -0
  78. rasa/cli/shell.py +141 -0
  79. rasa/cli/studio/__init__.py +0 -0
  80. rasa/cli/studio/download.py +62 -0
  81. rasa/cli/studio/studio.py +266 -0
  82. rasa/cli/studio/train.py +59 -0
  83. rasa/cli/studio/upload.py +77 -0
  84. rasa/cli/telemetry.py +102 -0
  85. rasa/cli/test.py +280 -0
  86. rasa/cli/train.py +260 -0
  87. rasa/cli/utils.py +464 -0
  88. rasa/cli/visualize.py +40 -0
  89. rasa/cli/x.py +206 -0
  90. rasa/constants.py +37 -0
  91. rasa/core/__init__.py +17 -0
  92. rasa/core/actions/__init__.py +0 -0
  93. rasa/core/actions/action.py +1225 -0
  94. rasa/core/actions/action_clean_stack.py +59 -0
  95. rasa/core/actions/action_exceptions.py +24 -0
  96. rasa/core/actions/action_run_slot_rejections.py +207 -0
  97. rasa/core/actions/action_trigger_chitchat.py +31 -0
  98. rasa/core/actions/action_trigger_flow.py +109 -0
  99. rasa/core/actions/action_trigger_search.py +31 -0
  100. rasa/core/actions/constants.py +5 -0
  101. rasa/core/actions/custom_action_executor.py +188 -0
  102. rasa/core/actions/forms.py +741 -0
  103. rasa/core/actions/grpc_custom_action_executor.py +251 -0
  104. rasa/core/actions/http_custom_action_executor.py +140 -0
  105. rasa/core/actions/loops.py +114 -0
  106. rasa/core/actions/two_stage_fallback.py +186 -0
  107. rasa/core/agent.py +555 -0
  108. rasa/core/auth_retry_tracker_store.py +122 -0
  109. rasa/core/brokers/__init__.py +0 -0
  110. rasa/core/brokers/broker.py +126 -0
  111. rasa/core/brokers/file.py +58 -0
  112. rasa/core/brokers/kafka.py +322 -0
  113. rasa/core/brokers/pika.py +386 -0
  114. rasa/core/brokers/sql.py +86 -0
  115. rasa/core/channels/__init__.py +55 -0
  116. rasa/core/channels/audiocodes.py +463 -0
  117. rasa/core/channels/botframework.py +338 -0
  118. rasa/core/channels/callback.py +84 -0
  119. rasa/core/channels/channel.py +419 -0
  120. rasa/core/channels/console.py +241 -0
  121. rasa/core/channels/development_inspector.py +93 -0
  122. rasa/core/channels/facebook.py +419 -0
  123. rasa/core/channels/hangouts.py +329 -0
  124. rasa/core/channels/inspector/.eslintrc.cjs +25 -0
  125. rasa/core/channels/inspector/.gitignore +23 -0
  126. rasa/core/channels/inspector/README.md +54 -0
  127. rasa/core/channels/inspector/assets/favicon.ico +0 -0
  128. rasa/core/channels/inspector/assets/rasa-chat.js +2 -0
  129. rasa/core/channels/inspector/custom.d.ts +3 -0
  130. rasa/core/channels/inspector/dist/assets/arc-b6e548fe.js +1 -0
  131. rasa/core/channels/inspector/dist/assets/array-9f3ba611.js +1 -0
  132. rasa/core/channels/inspector/dist/assets/c4Diagram-d0fbc5ce-fa03ac9e.js +10 -0
  133. rasa/core/channels/inspector/dist/assets/classDiagram-936ed81e-ee67392a.js +2 -0
  134. rasa/core/channels/inspector/dist/assets/classDiagram-v2-c3cb15f1-9b283fae.js +2 -0
  135. rasa/core/channels/inspector/dist/assets/createText-62fc7601-8b6fcc2a.js +7 -0
  136. rasa/core/channels/inspector/dist/assets/edges-f2ad444c-22e77f4f.js +4 -0
  137. rasa/core/channels/inspector/dist/assets/erDiagram-9d236eb7-60ffc87f.js +51 -0
  138. rasa/core/channels/inspector/dist/assets/flowDb-1972c806-9dd802e4.js +6 -0
  139. rasa/core/channels/inspector/dist/assets/flowDiagram-7ea5b25a-5fa1912f.js +4 -0
  140. rasa/core/channels/inspector/dist/assets/flowDiagram-v2-855bc5b3-1844e5a5.js +1 -0
  141. rasa/core/channels/inspector/dist/assets/flowchart-elk-definition-abe16c3d-622a1fd2.js +139 -0
  142. rasa/core/channels/inspector/dist/assets/ganttDiagram-9b5ea136-e285a63a.js +266 -0
  143. rasa/core/channels/inspector/dist/assets/gitGraphDiagram-99d0ae7c-f237bdca.js +70 -0
  144. rasa/core/channels/inspector/dist/assets/ibm-plex-mono-v4-latin-regular-128cfa44.ttf +0 -0
  145. rasa/core/channels/inspector/dist/assets/ibm-plex-mono-v4-latin-regular-21dbcb97.woff +0 -0
  146. rasa/core/channels/inspector/dist/assets/ibm-plex-mono-v4-latin-regular-222b5e26.svg +329 -0
  147. rasa/core/channels/inspector/dist/assets/ibm-plex-mono-v4-latin-regular-9ad89b2a.woff2 +0 -0
  148. rasa/core/channels/inspector/dist/assets/index-2c4b9a3b-4b03d70e.js +1 -0
  149. rasa/core/channels/inspector/dist/assets/index-3ee28881.css +1 -0
  150. rasa/core/channels/inspector/dist/assets/index-a5d3e69d.js +1040 -0
  151. rasa/core/channels/inspector/dist/assets/infoDiagram-736b4530-72a0fa5f.js +7 -0
  152. rasa/core/channels/inspector/dist/assets/init-77b53fdd.js +1 -0
  153. rasa/core/channels/inspector/dist/assets/journeyDiagram-df861f2b-82218c41.js +139 -0
  154. rasa/core/channels/inspector/dist/assets/lato-v14-latin-700-60c05ee4.woff +0 -0
  155. rasa/core/channels/inspector/dist/assets/lato-v14-latin-700-8335d9b8.svg +438 -0
  156. rasa/core/channels/inspector/dist/assets/lato-v14-latin-700-9cc39c75.ttf +0 -0
  157. rasa/core/channels/inspector/dist/assets/lato-v14-latin-700-ead13ccf.woff2 +0 -0
  158. rasa/core/channels/inspector/dist/assets/lato-v14-latin-regular-16705655.woff2 +0 -0
  159. rasa/core/channels/inspector/dist/assets/lato-v14-latin-regular-5aeb07f9.woff +0 -0
  160. rasa/core/channels/inspector/dist/assets/lato-v14-latin-regular-9c459044.ttf +0 -0
  161. rasa/core/channels/inspector/dist/assets/lato-v14-latin-regular-9e2898a4.svg +435 -0
  162. rasa/core/channels/inspector/dist/assets/layout-78cff630.js +1 -0
  163. rasa/core/channels/inspector/dist/assets/line-5038b469.js +1 -0
  164. rasa/core/channels/inspector/dist/assets/linear-c4fc4098.js +1 -0
  165. rasa/core/channels/inspector/dist/assets/mindmap-definition-beec6740-c33c8ea6.js +109 -0
  166. rasa/core/channels/inspector/dist/assets/ordinal-ba9b4969.js +1 -0
  167. rasa/core/channels/inspector/dist/assets/path-53f90ab3.js +1 -0
  168. rasa/core/channels/inspector/dist/assets/pieDiagram-dbbf0591-a8d03059.js +35 -0
  169. rasa/core/channels/inspector/dist/assets/quadrantDiagram-4d7f4fd6-6a0e56b2.js +7 -0
  170. rasa/core/channels/inspector/dist/assets/requirementDiagram-6fc4c22a-2dc7c7bd.js +52 -0
  171. rasa/core/channels/inspector/dist/assets/sankeyDiagram-8f13d901-2360fe39.js +8 -0
  172. rasa/core/channels/inspector/dist/assets/sequenceDiagram-b655622a-41b9f9ad.js +122 -0
  173. rasa/core/channels/inspector/dist/assets/stateDiagram-59f0c015-0aad326f.js +1 -0
  174. rasa/core/channels/inspector/dist/assets/stateDiagram-v2-2b26beab-9847d984.js +1 -0
  175. rasa/core/channels/inspector/dist/assets/styles-080da4f6-564d890e.js +110 -0
  176. rasa/core/channels/inspector/dist/assets/styles-3dcbcfbf-38957613.js +159 -0
  177. rasa/core/channels/inspector/dist/assets/styles-9c745c82-f0fc6921.js +207 -0
  178. rasa/core/channels/inspector/dist/assets/svgDrawCommon-4835440b-ef3c5a77.js +1 -0
  179. rasa/core/channels/inspector/dist/assets/timeline-definition-5b62e21b-bf3e91c1.js +61 -0
  180. rasa/core/channels/inspector/dist/assets/xychartDiagram-2b33534f-4d4026c0.js +7 -0
  181. rasa/core/channels/inspector/dist/index.html +41 -0
  182. rasa/core/channels/inspector/index.html +39 -0
  183. rasa/core/channels/inspector/jest.config.ts +13 -0
  184. rasa/core/channels/inspector/package.json +48 -0
  185. rasa/core/channels/inspector/setupTests.ts +2 -0
  186. rasa/core/channels/inspector/src/App.tsx +170 -0
  187. rasa/core/channels/inspector/src/components/DiagramFlow.tsx +107 -0
  188. rasa/core/channels/inspector/src/components/DialogueInformation.tsx +187 -0
  189. rasa/core/channels/inspector/src/components/DialogueStack.tsx +151 -0
  190. rasa/core/channels/inspector/src/components/ExpandIcon.tsx +16 -0
  191. rasa/core/channels/inspector/src/components/FullscreenButton.tsx +45 -0
  192. rasa/core/channels/inspector/src/components/LoadingSpinner.tsx +19 -0
  193. rasa/core/channels/inspector/src/components/NoActiveFlow.tsx +21 -0
  194. rasa/core/channels/inspector/src/components/RasaLogo.tsx +32 -0
  195. rasa/core/channels/inspector/src/components/SaraDiagrams.tsx +39 -0
  196. rasa/core/channels/inspector/src/components/Slots.tsx +91 -0
  197. rasa/core/channels/inspector/src/components/Welcome.tsx +54 -0
  198. rasa/core/channels/inspector/src/helpers/formatters.test.ts +382 -0
  199. rasa/core/channels/inspector/src/helpers/formatters.ts +240 -0
  200. rasa/core/channels/inspector/src/helpers/utils.ts +42 -0
  201. rasa/core/channels/inspector/src/main.tsx +13 -0
  202. rasa/core/channels/inspector/src/theme/Button/Button.ts +29 -0
  203. rasa/core/channels/inspector/src/theme/Heading/Heading.ts +31 -0
  204. rasa/core/channels/inspector/src/theme/Input/Input.ts +27 -0
  205. rasa/core/channels/inspector/src/theme/Link/Link.ts +10 -0
  206. rasa/core/channels/inspector/src/theme/Modal/Modal.ts +47 -0
  207. rasa/core/channels/inspector/src/theme/Table/Table.tsx +38 -0
  208. rasa/core/channels/inspector/src/theme/Tooltip/Tooltip.ts +12 -0
  209. rasa/core/channels/inspector/src/theme/base/breakpoints.ts +8 -0
  210. rasa/core/channels/inspector/src/theme/base/colors.ts +88 -0
  211. rasa/core/channels/inspector/src/theme/base/fonts/fontFaces.css +29 -0
  212. rasa/core/channels/inspector/src/theme/base/fonts/ibm-plex-mono-v4-latin/ibm-plex-mono-v4-latin-regular.eot +0 -0
  213. rasa/core/channels/inspector/src/theme/base/fonts/ibm-plex-mono-v4-latin/ibm-plex-mono-v4-latin-regular.svg +329 -0
  214. rasa/core/channels/inspector/src/theme/base/fonts/ibm-plex-mono-v4-latin/ibm-plex-mono-v4-latin-regular.ttf +0 -0
  215. rasa/core/channels/inspector/src/theme/base/fonts/ibm-plex-mono-v4-latin/ibm-plex-mono-v4-latin-regular.woff +0 -0
  216. rasa/core/channels/inspector/src/theme/base/fonts/ibm-plex-mono-v4-latin/ibm-plex-mono-v4-latin-regular.woff2 +0 -0
  217. rasa/core/channels/inspector/src/theme/base/fonts/lato-v14-latin/lato-v14-latin-700.eot +0 -0
  218. rasa/core/channels/inspector/src/theme/base/fonts/lato-v14-latin/lato-v14-latin-700.svg +438 -0
  219. rasa/core/channels/inspector/src/theme/base/fonts/lato-v14-latin/lato-v14-latin-700.ttf +0 -0
  220. rasa/core/channels/inspector/src/theme/base/fonts/lato-v14-latin/lato-v14-latin-700.woff +0 -0
  221. rasa/core/channels/inspector/src/theme/base/fonts/lato-v14-latin/lato-v14-latin-700.woff2 +0 -0
  222. rasa/core/channels/inspector/src/theme/base/fonts/lato-v14-latin/lato-v14-latin-regular.eot +0 -0
  223. rasa/core/channels/inspector/src/theme/base/fonts/lato-v14-latin/lato-v14-latin-regular.svg +435 -0
  224. rasa/core/channels/inspector/src/theme/base/fonts/lato-v14-latin/lato-v14-latin-regular.ttf +0 -0
  225. rasa/core/channels/inspector/src/theme/base/fonts/lato-v14-latin/lato-v14-latin-regular.woff +0 -0
  226. rasa/core/channels/inspector/src/theme/base/fonts/lato-v14-latin/lato-v14-latin-regular.woff2 +0 -0
  227. rasa/core/channels/inspector/src/theme/base/radii.ts +9 -0
  228. rasa/core/channels/inspector/src/theme/base/shadows.ts +7 -0
  229. rasa/core/channels/inspector/src/theme/base/sizes.ts +7 -0
  230. rasa/core/channels/inspector/src/theme/base/space.ts +15 -0
  231. rasa/core/channels/inspector/src/theme/base/styles.ts +13 -0
  232. rasa/core/channels/inspector/src/theme/base/typography.ts +24 -0
  233. rasa/core/channels/inspector/src/theme/base/zIndices.ts +19 -0
  234. rasa/core/channels/inspector/src/theme/index.ts +101 -0
  235. rasa/core/channels/inspector/src/types.ts +64 -0
  236. rasa/core/channels/inspector/src/vite-env.d.ts +1 -0
  237. rasa/core/channels/inspector/tests/__mocks__/fileMock.ts +1 -0
  238. rasa/core/channels/inspector/tests/__mocks__/matchMedia.ts +16 -0
  239. rasa/core/channels/inspector/tests/__mocks__/styleMock.ts +1 -0
  240. rasa/core/channels/inspector/tests/renderWithProviders.tsx +14 -0
  241. rasa/core/channels/inspector/tsconfig.json +26 -0
  242. rasa/core/channels/inspector/tsconfig.node.json +10 -0
  243. rasa/core/channels/inspector/vite.config.ts +8 -0
  244. rasa/core/channels/inspector/yarn.lock +6156 -0
  245. rasa/core/channels/mattermost.py +229 -0
  246. rasa/core/channels/rasa_chat.py +126 -0
  247. rasa/core/channels/rest.py +225 -0
  248. rasa/core/channels/rocketchat.py +174 -0
  249. rasa/core/channels/slack.py +620 -0
  250. rasa/core/channels/socketio.py +274 -0
  251. rasa/core/channels/telegram.py +298 -0
  252. rasa/core/channels/twilio.py +169 -0
  253. rasa/core/channels/twilio_voice.py +367 -0
  254. rasa/core/channels/vier_cvg.py +374 -0
  255. rasa/core/channels/webexteams.py +134 -0
  256. rasa/core/concurrent_lock_store.py +210 -0
  257. rasa/core/constants.py +107 -0
  258. rasa/core/evaluation/__init__.py +0 -0
  259. rasa/core/evaluation/marker.py +267 -0
  260. rasa/core/evaluation/marker_base.py +923 -0
  261. rasa/core/evaluation/marker_stats.py +293 -0
  262. rasa/core/evaluation/marker_tracker_loader.py +103 -0
  263. rasa/core/exceptions.py +29 -0
  264. rasa/core/exporter.py +284 -0
  265. rasa/core/featurizers/__init__.py +0 -0
  266. rasa/core/featurizers/precomputation.py +410 -0
  267. rasa/core/featurizers/single_state_featurizer.py +421 -0
  268. rasa/core/featurizers/tracker_featurizers.py +1262 -0
  269. rasa/core/http_interpreter.py +89 -0
  270. rasa/core/information_retrieval/__init__.py +7 -0
  271. rasa/core/information_retrieval/faiss.py +121 -0
  272. rasa/core/information_retrieval/information_retrieval.py +129 -0
  273. rasa/core/information_retrieval/milvus.py +52 -0
  274. rasa/core/information_retrieval/qdrant.py +95 -0
  275. rasa/core/jobs.py +63 -0
  276. rasa/core/lock.py +139 -0
  277. rasa/core/lock_store.py +343 -0
  278. rasa/core/migrate.py +403 -0
  279. rasa/core/nlg/__init__.py +3 -0
  280. rasa/core/nlg/callback.py +146 -0
  281. rasa/core/nlg/contextual_response_rephraser.py +270 -0
  282. rasa/core/nlg/generator.py +230 -0
  283. rasa/core/nlg/interpolator.py +143 -0
  284. rasa/core/nlg/response.py +155 -0
  285. rasa/core/nlg/summarize.py +69 -0
  286. rasa/core/policies/__init__.py +0 -0
  287. rasa/core/policies/ensemble.py +329 -0
  288. rasa/core/policies/enterprise_search_policy.py +781 -0
  289. rasa/core/policies/enterprise_search_prompt_template.jinja2 +25 -0
  290. rasa/core/policies/enterprise_search_prompt_with_citation_template.jinja2 +60 -0
  291. rasa/core/policies/flow_policy.py +205 -0
  292. rasa/core/policies/flows/__init__.py +0 -0
  293. rasa/core/policies/flows/flow_exceptions.py +44 -0
  294. rasa/core/policies/flows/flow_executor.py +705 -0
  295. rasa/core/policies/flows/flow_step_result.py +43 -0
  296. rasa/core/policies/intentless_policy.py +922 -0
  297. rasa/core/policies/intentless_prompt_template.jinja2 +22 -0
  298. rasa/core/policies/memoization.py +538 -0
  299. rasa/core/policies/policy.py +725 -0
  300. rasa/core/policies/rule_policy.py +1273 -0
  301. rasa/core/policies/ted_policy.py +2169 -0
  302. rasa/core/policies/unexpected_intent_policy.py +1022 -0
  303. rasa/core/processor.py +1422 -0
  304. rasa/core/run.py +331 -0
  305. rasa/core/secrets_manager/__init__.py +0 -0
  306. rasa/core/secrets_manager/constants.py +32 -0
  307. rasa/core/secrets_manager/endpoints.py +391 -0
  308. rasa/core/secrets_manager/factory.py +233 -0
  309. rasa/core/secrets_manager/secret_manager.py +262 -0
  310. rasa/core/secrets_manager/vault.py +574 -0
  311. rasa/core/test.py +1335 -0
  312. rasa/core/tracker_store.py +1699 -0
  313. rasa/core/train.py +105 -0
  314. rasa/core/training/__init__.py +89 -0
  315. rasa/core/training/converters/__init__.py +0 -0
  316. rasa/core/training/converters/responses_prefix_converter.py +119 -0
  317. rasa/core/training/interactive.py +1745 -0
  318. rasa/core/training/story_conflict.py +381 -0
  319. rasa/core/training/training.py +93 -0
  320. rasa/core/utils.py +339 -0
  321. rasa/core/visualize.py +70 -0
  322. rasa/dialogue_understanding/__init__.py +0 -0
  323. rasa/dialogue_understanding/coexistence/__init__.py +0 -0
  324. rasa/dialogue_understanding/coexistence/constants.py +4 -0
  325. rasa/dialogue_understanding/coexistence/intent_based_router.py +196 -0
  326. rasa/dialogue_understanding/coexistence/llm_based_router.py +260 -0
  327. rasa/dialogue_understanding/coexistence/router_template.jinja2 +12 -0
  328. rasa/dialogue_understanding/commands/__init__.py +49 -0
  329. rasa/dialogue_understanding/commands/can_not_handle_command.py +70 -0
  330. rasa/dialogue_understanding/commands/cancel_flow_command.py +125 -0
  331. rasa/dialogue_understanding/commands/change_flow_command.py +44 -0
  332. rasa/dialogue_understanding/commands/chit_chat_answer_command.py +57 -0
  333. rasa/dialogue_understanding/commands/clarify_command.py +86 -0
  334. rasa/dialogue_understanding/commands/command.py +85 -0
  335. rasa/dialogue_understanding/commands/correct_slots_command.py +297 -0
  336. rasa/dialogue_understanding/commands/error_command.py +79 -0
  337. rasa/dialogue_understanding/commands/free_form_answer_command.py +9 -0
  338. rasa/dialogue_understanding/commands/handle_code_change_command.py +73 -0
  339. rasa/dialogue_understanding/commands/human_handoff_command.py +66 -0
  340. rasa/dialogue_understanding/commands/knowledge_answer_command.py +57 -0
  341. rasa/dialogue_understanding/commands/noop_command.py +54 -0
  342. rasa/dialogue_understanding/commands/set_slot_command.py +160 -0
  343. rasa/dialogue_understanding/commands/skip_question_command.py +75 -0
  344. rasa/dialogue_understanding/commands/start_flow_command.py +107 -0
  345. rasa/dialogue_understanding/generator/__init__.py +21 -0
  346. rasa/dialogue_understanding/generator/command_generator.py +343 -0
  347. rasa/dialogue_understanding/generator/constants.py +18 -0
  348. rasa/dialogue_understanding/generator/flow_document_template.jinja2 +4 -0
  349. rasa/dialogue_understanding/generator/flow_retrieval.py +412 -0
  350. rasa/dialogue_understanding/generator/llm_based_command_generator.py +467 -0
  351. rasa/dialogue_understanding/generator/llm_command_generator.py +67 -0
  352. rasa/dialogue_understanding/generator/multi_step/__init__.py +0 -0
  353. rasa/dialogue_understanding/generator/multi_step/fill_slots_prompt.jinja2 +62 -0
  354. rasa/dialogue_understanding/generator/multi_step/handle_flows_prompt.jinja2 +38 -0
  355. rasa/dialogue_understanding/generator/multi_step/multi_step_llm_command_generator.py +827 -0
  356. rasa/dialogue_understanding/generator/nlu_command_adapter.py +218 -0
  357. rasa/dialogue_understanding/generator/single_step/__init__.py +0 -0
  358. rasa/dialogue_understanding/generator/single_step/command_prompt_template.jinja2 +57 -0
  359. rasa/dialogue_understanding/generator/single_step/single_step_llm_command_generator.py +345 -0
  360. rasa/dialogue_understanding/patterns/__init__.py +0 -0
  361. rasa/dialogue_understanding/patterns/cancel.py +111 -0
  362. rasa/dialogue_understanding/patterns/cannot_handle.py +43 -0
  363. rasa/dialogue_understanding/patterns/chitchat.py +37 -0
  364. rasa/dialogue_understanding/patterns/clarify.py +97 -0
  365. rasa/dialogue_understanding/patterns/code_change.py +41 -0
  366. rasa/dialogue_understanding/patterns/collect_information.py +90 -0
  367. rasa/dialogue_understanding/patterns/completed.py +40 -0
  368. rasa/dialogue_understanding/patterns/continue_interrupted.py +42 -0
  369. rasa/dialogue_understanding/patterns/correction.py +278 -0
  370. rasa/dialogue_understanding/patterns/default_flows_for_patterns.yml +248 -0
  371. rasa/dialogue_understanding/patterns/human_handoff.py +37 -0
  372. rasa/dialogue_understanding/patterns/internal_error.py +47 -0
  373. rasa/dialogue_understanding/patterns/search.py +37 -0
  374. rasa/dialogue_understanding/patterns/skip_question.py +38 -0
  375. rasa/dialogue_understanding/processor/__init__.py +0 -0
  376. rasa/dialogue_understanding/processor/command_processor.py +687 -0
  377. rasa/dialogue_understanding/processor/command_processor_component.py +39 -0
  378. rasa/dialogue_understanding/stack/__init__.py +0 -0
  379. rasa/dialogue_understanding/stack/dialogue_stack.py +178 -0
  380. rasa/dialogue_understanding/stack/frames/__init__.py +19 -0
  381. rasa/dialogue_understanding/stack/frames/chit_chat_frame.py +27 -0
  382. rasa/dialogue_understanding/stack/frames/dialogue_stack_frame.py +137 -0
  383. rasa/dialogue_understanding/stack/frames/flow_stack_frame.py +157 -0
  384. rasa/dialogue_understanding/stack/frames/pattern_frame.py +10 -0
  385. rasa/dialogue_understanding/stack/frames/search_frame.py +27 -0
  386. rasa/dialogue_understanding/stack/utils.py +211 -0
  387. rasa/e2e_test/__init__.py +0 -0
  388. rasa/e2e_test/constants.py +11 -0
  389. rasa/e2e_test/e2e_test_case.py +366 -0
  390. rasa/e2e_test/e2e_test_result.py +34 -0
  391. rasa/e2e_test/e2e_test_runner.py +768 -0
  392. rasa/e2e_test/e2e_test_schema.yml +85 -0
  393. rasa/engine/__init__.py +0 -0
  394. rasa/engine/caching.py +463 -0
  395. rasa/engine/constants.py +17 -0
  396. rasa/engine/exceptions.py +14 -0
  397. rasa/engine/graph.py +637 -0
  398. rasa/engine/loader.py +36 -0
  399. rasa/engine/recipes/__init__.py +0 -0
  400. rasa/engine/recipes/config_files/default_config.yml +44 -0
  401. rasa/engine/recipes/default_components.py +99 -0
  402. rasa/engine/recipes/default_recipe.py +1251 -0
  403. rasa/engine/recipes/graph_recipe.py +79 -0
  404. rasa/engine/recipes/recipe.py +93 -0
  405. rasa/engine/runner/__init__.py +0 -0
  406. rasa/engine/runner/dask.py +250 -0
  407. rasa/engine/runner/interface.py +49 -0
  408. rasa/engine/storage/__init__.py +0 -0
  409. rasa/engine/storage/local_model_storage.py +246 -0
  410. rasa/engine/storage/resource.py +110 -0
  411. rasa/engine/storage/storage.py +203 -0
  412. rasa/engine/training/__init__.py +0 -0
  413. rasa/engine/training/components.py +176 -0
  414. rasa/engine/training/fingerprinting.py +64 -0
  415. rasa/engine/training/graph_trainer.py +256 -0
  416. rasa/engine/training/hooks.py +164 -0
  417. rasa/engine/validation.py +873 -0
  418. rasa/env.py +5 -0
  419. rasa/exceptions.py +69 -0
  420. rasa/graph_components/__init__.py +0 -0
  421. rasa/graph_components/converters/__init__.py +0 -0
  422. rasa/graph_components/converters/nlu_message_converter.py +48 -0
  423. rasa/graph_components/providers/__init__.py +0 -0
  424. rasa/graph_components/providers/domain_for_core_training_provider.py +87 -0
  425. rasa/graph_components/providers/domain_provider.py +71 -0
  426. rasa/graph_components/providers/flows_provider.py +74 -0
  427. rasa/graph_components/providers/forms_provider.py +44 -0
  428. rasa/graph_components/providers/nlu_training_data_provider.py +56 -0
  429. rasa/graph_components/providers/responses_provider.py +44 -0
  430. rasa/graph_components/providers/rule_only_provider.py +49 -0
  431. rasa/graph_components/providers/story_graph_provider.py +43 -0
  432. rasa/graph_components/providers/training_tracker_provider.py +55 -0
  433. rasa/graph_components/validators/__init__.py +0 -0
  434. rasa/graph_components/validators/default_recipe_validator.py +550 -0
  435. rasa/graph_components/validators/finetuning_validator.py +302 -0
  436. rasa/hooks.py +112 -0
  437. rasa/jupyter.py +63 -0
  438. rasa/markers/__init__.py +0 -0
  439. rasa/markers/marker.py +269 -0
  440. rasa/markers/marker_base.py +828 -0
  441. rasa/markers/upload.py +74 -0
  442. rasa/markers/validate.py +21 -0
  443. rasa/model.py +118 -0
  444. rasa/model_testing.py +457 -0
  445. rasa/model_training.py +536 -0
  446. rasa/nlu/__init__.py +7 -0
  447. rasa/nlu/classifiers/__init__.py +3 -0
  448. rasa/nlu/classifiers/classifier.py +5 -0
  449. rasa/nlu/classifiers/diet_classifier.py +1881 -0
  450. rasa/nlu/classifiers/fallback_classifier.py +192 -0
  451. rasa/nlu/classifiers/keyword_intent_classifier.py +188 -0
  452. rasa/nlu/classifiers/llm_intent_classifier.py +519 -0
  453. rasa/nlu/classifiers/logistic_regression_classifier.py +253 -0
  454. rasa/nlu/classifiers/mitie_intent_classifier.py +156 -0
  455. rasa/nlu/classifiers/regex_message_handler.py +56 -0
  456. rasa/nlu/classifiers/sklearn_intent_classifier.py +330 -0
  457. rasa/nlu/constants.py +77 -0
  458. rasa/nlu/convert.py +40 -0
  459. rasa/nlu/emulators/__init__.py +0 -0
  460. rasa/nlu/emulators/dialogflow.py +55 -0
  461. rasa/nlu/emulators/emulator.py +49 -0
  462. rasa/nlu/emulators/luis.py +86 -0
  463. rasa/nlu/emulators/no_emulator.py +10 -0
  464. rasa/nlu/emulators/wit.py +56 -0
  465. rasa/nlu/extractors/__init__.py +0 -0
  466. rasa/nlu/extractors/crf_entity_extractor.py +715 -0
  467. rasa/nlu/extractors/duckling_entity_extractor.py +206 -0
  468. rasa/nlu/extractors/entity_synonyms.py +178 -0
  469. rasa/nlu/extractors/extractor.py +470 -0
  470. rasa/nlu/extractors/mitie_entity_extractor.py +293 -0
  471. rasa/nlu/extractors/regex_entity_extractor.py +220 -0
  472. rasa/nlu/extractors/spacy_entity_extractor.py +95 -0
  473. rasa/nlu/featurizers/__init__.py +0 -0
  474. rasa/nlu/featurizers/dense_featurizer/__init__.py +0 -0
  475. rasa/nlu/featurizers/dense_featurizer/convert_featurizer.py +445 -0
  476. rasa/nlu/featurizers/dense_featurizer/dense_featurizer.py +57 -0
  477. rasa/nlu/featurizers/dense_featurizer/lm_featurizer.py +768 -0
  478. rasa/nlu/featurizers/dense_featurizer/mitie_featurizer.py +170 -0
  479. rasa/nlu/featurizers/dense_featurizer/spacy_featurizer.py +132 -0
  480. rasa/nlu/featurizers/featurizer.py +89 -0
  481. rasa/nlu/featurizers/sparse_featurizer/__init__.py +0 -0
  482. rasa/nlu/featurizers/sparse_featurizer/count_vectors_featurizer.py +867 -0
  483. rasa/nlu/featurizers/sparse_featurizer/lexical_syntactic_featurizer.py +571 -0
  484. rasa/nlu/featurizers/sparse_featurizer/regex_featurizer.py +271 -0
  485. rasa/nlu/featurizers/sparse_featurizer/sparse_featurizer.py +9 -0
  486. rasa/nlu/model.py +24 -0
  487. rasa/nlu/persistor.py +282 -0
  488. rasa/nlu/run.py +27 -0
  489. rasa/nlu/selectors/__init__.py +0 -0
  490. rasa/nlu/selectors/response_selector.py +987 -0
  491. rasa/nlu/test.py +1940 -0
  492. rasa/nlu/tokenizers/__init__.py +0 -0
  493. rasa/nlu/tokenizers/jieba_tokenizer.py +148 -0
  494. rasa/nlu/tokenizers/mitie_tokenizer.py +75 -0
  495. rasa/nlu/tokenizers/spacy_tokenizer.py +72 -0
  496. rasa/nlu/tokenizers/tokenizer.py +239 -0
  497. rasa/nlu/tokenizers/whitespace_tokenizer.py +106 -0
  498. rasa/nlu/utils/__init__.py +35 -0
  499. rasa/nlu/utils/bilou_utils.py +462 -0
  500. rasa/nlu/utils/hugging_face/__init__.py +0 -0
  501. rasa/nlu/utils/hugging_face/registry.py +108 -0
  502. rasa/nlu/utils/hugging_face/transformers_pre_post_processors.py +311 -0
  503. rasa/nlu/utils/mitie_utils.py +113 -0
  504. rasa/nlu/utils/pattern_utils.py +168 -0
  505. rasa/nlu/utils/spacy_utils.py +310 -0
  506. rasa/plugin.py +90 -0
  507. rasa/server.py +1551 -0
  508. rasa/shared/__init__.py +0 -0
  509. rasa/shared/constants.py +192 -0
  510. rasa/shared/core/__init__.py +0 -0
  511. rasa/shared/core/command_payload_reader.py +109 -0
  512. rasa/shared/core/constants.py +167 -0
  513. rasa/shared/core/conversation.py +46 -0
  514. rasa/shared/core/domain.py +2107 -0
  515. rasa/shared/core/events.py +2504 -0
  516. rasa/shared/core/flows/__init__.py +7 -0
  517. rasa/shared/core/flows/flow.py +362 -0
  518. rasa/shared/core/flows/flow_step.py +146 -0
  519. rasa/shared/core/flows/flow_step_links.py +319 -0
  520. rasa/shared/core/flows/flow_step_sequence.py +70 -0
  521. rasa/shared/core/flows/flows_list.py +223 -0
  522. rasa/shared/core/flows/flows_yaml_schema.json +217 -0
  523. rasa/shared/core/flows/nlu_trigger.py +117 -0
  524. rasa/shared/core/flows/steps/__init__.py +24 -0
  525. rasa/shared/core/flows/steps/action.py +56 -0
  526. rasa/shared/core/flows/steps/call.py +64 -0
  527. rasa/shared/core/flows/steps/collect.py +112 -0
  528. rasa/shared/core/flows/steps/constants.py +5 -0
  529. rasa/shared/core/flows/steps/continuation.py +36 -0
  530. rasa/shared/core/flows/steps/end.py +22 -0
  531. rasa/shared/core/flows/steps/internal.py +44 -0
  532. rasa/shared/core/flows/steps/link.py +51 -0
  533. rasa/shared/core/flows/steps/no_operation.py +48 -0
  534. rasa/shared/core/flows/steps/set_slots.py +50 -0
  535. rasa/shared/core/flows/steps/start.py +30 -0
  536. rasa/shared/core/flows/validation.py +527 -0
  537. rasa/shared/core/flows/yaml_flows_io.py +278 -0
  538. rasa/shared/core/generator.py +908 -0
  539. rasa/shared/core/slot_mappings.py +526 -0
  540. rasa/shared/core/slots.py +649 -0
  541. rasa/shared/core/trackers.py +1177 -0
  542. rasa/shared/core/training_data/__init__.py +0 -0
  543. rasa/shared/core/training_data/loading.py +89 -0
  544. rasa/shared/core/training_data/story_reader/__init__.py +0 -0
  545. rasa/shared/core/training_data/story_reader/story_reader.py +129 -0
  546. rasa/shared/core/training_data/story_reader/story_step_builder.py +168 -0
  547. rasa/shared/core/training_data/story_reader/yaml_story_reader.py +888 -0
  548. rasa/shared/core/training_data/story_writer/__init__.py +0 -0
  549. rasa/shared/core/training_data/story_writer/story_writer.py +76 -0
  550. rasa/shared/core/training_data/story_writer/yaml_story_writer.py +444 -0
  551. rasa/shared/core/training_data/structures.py +838 -0
  552. rasa/shared/core/training_data/visualization.html +146 -0
  553. rasa/shared/core/training_data/visualization.py +603 -0
  554. rasa/shared/data.py +249 -0
  555. rasa/shared/engine/__init__.py +0 -0
  556. rasa/shared/engine/caching.py +26 -0
  557. rasa/shared/exceptions.py +163 -0
  558. rasa/shared/importers/__init__.py +0 -0
  559. rasa/shared/importers/importer.py +704 -0
  560. rasa/shared/importers/multi_project.py +203 -0
  561. rasa/shared/importers/rasa.py +99 -0
  562. rasa/shared/importers/utils.py +34 -0
  563. rasa/shared/nlu/__init__.py +0 -0
  564. rasa/shared/nlu/constants.py +47 -0
  565. rasa/shared/nlu/interpreter.py +10 -0
  566. rasa/shared/nlu/training_data/__init__.py +0 -0
  567. rasa/shared/nlu/training_data/entities_parser.py +208 -0
  568. rasa/shared/nlu/training_data/features.py +492 -0
  569. rasa/shared/nlu/training_data/formats/__init__.py +10 -0
  570. rasa/shared/nlu/training_data/formats/dialogflow.py +163 -0
  571. rasa/shared/nlu/training_data/formats/luis.py +87 -0
  572. rasa/shared/nlu/training_data/formats/rasa.py +135 -0
  573. rasa/shared/nlu/training_data/formats/rasa_yaml.py +603 -0
  574. rasa/shared/nlu/training_data/formats/readerwriter.py +244 -0
  575. rasa/shared/nlu/training_data/formats/wit.py +52 -0
  576. rasa/shared/nlu/training_data/loading.py +137 -0
  577. rasa/shared/nlu/training_data/lookup_tables_parser.py +30 -0
  578. rasa/shared/nlu/training_data/message.py +490 -0
  579. rasa/shared/nlu/training_data/schemas/__init__.py +0 -0
  580. rasa/shared/nlu/training_data/schemas/data_schema.py +85 -0
  581. rasa/shared/nlu/training_data/schemas/nlu.yml +53 -0
  582. rasa/shared/nlu/training_data/schemas/responses.yml +70 -0
  583. rasa/shared/nlu/training_data/synonyms_parser.py +42 -0
  584. rasa/shared/nlu/training_data/training_data.py +730 -0
  585. rasa/shared/nlu/training_data/util.py +223 -0
  586. rasa/shared/providers/__init__.py +0 -0
  587. rasa/shared/providers/openai/__init__.py +0 -0
  588. rasa/shared/providers/openai/clients.py +43 -0
  589. rasa/shared/providers/openai/session_handler.py +110 -0
  590. rasa/shared/utils/__init__.py +0 -0
  591. rasa/shared/utils/cli.py +72 -0
  592. rasa/shared/utils/common.py +308 -0
  593. rasa/shared/utils/constants.py +4 -0
  594. rasa/shared/utils/io.py +415 -0
  595. rasa/shared/utils/llm.py +404 -0
  596. rasa/shared/utils/pykwalify_extensions.py +27 -0
  597. rasa/shared/utils/schemas/__init__.py +0 -0
  598. rasa/shared/utils/schemas/config.yml +2 -0
  599. rasa/shared/utils/schemas/domain.yml +145 -0
  600. rasa/shared/utils/schemas/events.py +212 -0
  601. rasa/shared/utils/schemas/model_config.yml +46 -0
  602. rasa/shared/utils/schemas/stories.yml +173 -0
  603. rasa/shared/utils/yaml.py +786 -0
  604. rasa/studio/__init__.py +0 -0
  605. rasa/studio/auth.py +268 -0
  606. rasa/studio/config.py +127 -0
  607. rasa/studio/constants.py +18 -0
  608. rasa/studio/data_handler.py +359 -0
  609. rasa/studio/download.py +483 -0
  610. rasa/studio/results_logger.py +137 -0
  611. rasa/studio/train.py +135 -0
  612. rasa/studio/upload.py +433 -0
  613. rasa/telemetry.py +1737 -0
  614. rasa/tracing/__init__.py +0 -0
  615. rasa/tracing/config.py +353 -0
  616. rasa/tracing/constants.py +62 -0
  617. rasa/tracing/instrumentation/__init__.py +0 -0
  618. rasa/tracing/instrumentation/attribute_extractors.py +672 -0
  619. rasa/tracing/instrumentation/instrumentation.py +1185 -0
  620. rasa/tracing/instrumentation/intentless_policy_instrumentation.py +144 -0
  621. rasa/tracing/instrumentation/metrics.py +294 -0
  622. rasa/tracing/metric_instrument_provider.py +205 -0
  623. rasa/utils/__init__.py +0 -0
  624. rasa/utils/beta.py +83 -0
  625. rasa/utils/cli.py +28 -0
  626. rasa/utils/common.py +635 -0
  627. rasa/utils/converter.py +53 -0
  628. rasa/utils/endpoints.py +302 -0
  629. rasa/utils/io.py +260 -0
  630. rasa/utils/licensing.py +534 -0
  631. rasa/utils/log_utils.py +174 -0
  632. rasa/utils/mapper.py +210 -0
  633. rasa/utils/ml_utils.py +145 -0
  634. rasa/utils/plotting.py +362 -0
  635. rasa/utils/singleton.py +23 -0
  636. rasa/utils/tensorflow/__init__.py +0 -0
  637. rasa/utils/tensorflow/callback.py +112 -0
  638. rasa/utils/tensorflow/constants.py +116 -0
  639. rasa/utils/tensorflow/crf.py +492 -0
  640. rasa/utils/tensorflow/data_generator.py +440 -0
  641. rasa/utils/tensorflow/environment.py +161 -0
  642. rasa/utils/tensorflow/exceptions.py +5 -0
  643. rasa/utils/tensorflow/feature_array.py +366 -0
  644. rasa/utils/tensorflow/layers.py +1565 -0
  645. rasa/utils/tensorflow/layers_utils.py +113 -0
  646. rasa/utils/tensorflow/metrics.py +281 -0
  647. rasa/utils/tensorflow/model_data.py +798 -0
  648. rasa/utils/tensorflow/model_data_utils.py +499 -0
  649. rasa/utils/tensorflow/models.py +935 -0
  650. rasa/utils/tensorflow/rasa_layers.py +1094 -0
  651. rasa/utils/tensorflow/transformer.py +640 -0
  652. rasa/utils/tensorflow/types.py +6 -0
  653. rasa/utils/train_utils.py +572 -0
  654. rasa/utils/url_tools.py +53 -0
  655. rasa/utils/yaml.py +54 -0
  656. rasa/validator.py +1337 -0
  657. rasa/version.py +3 -0
  658. rasa_pro-3.9.18.dist-info/METADATA +563 -0
  659. rasa_pro-3.9.18.dist-info/NOTICE +5 -0
  660. rasa_pro-3.9.18.dist-info/RECORD +662 -0
  661. rasa_pro-3.9.18.dist-info/WHEEL +4 -0
  662. rasa_pro-3.9.18.dist-info/entry_points.txt +3 -0
@@ -0,0 +1,1177 @@
1
+ import copy
2
+ import dataclasses
3
+ import itertools
4
+ import logging
5
+ import os
6
+ import time
7
+ from collections import deque
8
+ from enum import Enum
9
+ from typing import (
10
+ Dict,
11
+ Text,
12
+ Any,
13
+ Optional,
14
+ Iterator,
15
+ Generator,
16
+ Type,
17
+ TypeVar,
18
+ List,
19
+ Deque,
20
+ Iterable,
21
+ Union,
22
+ FrozenSet,
23
+ Tuple,
24
+ TYPE_CHECKING,
25
+ cast,
26
+ )
27
+
28
+ import rasa.shared.utils.io
29
+ from rasa.shared.constants import (
30
+ ASSISTANT_ID_KEY,
31
+ DEFAULT_SENDER_ID,
32
+ ROUTE_TO_CALM_SLOT,
33
+ )
34
+ from rasa.shared.core import events
35
+ from rasa.shared.core.constants import (
36
+ ACTION_LISTEN_NAME,
37
+ LOOP_NAME,
38
+ SHOULD_NOT_BE_SET,
39
+ PREVIOUS_ACTION,
40
+ ACTIVE_LOOP,
41
+ ACTION_SESSION_START_NAME,
42
+ FOLLOWUP_ACTION,
43
+ )
44
+ from rasa.shared.core.conversation import Dialogue
45
+ from rasa.shared.core.domain import Domain, State
46
+ from rasa.shared.core.events import (
47
+ DialogueStackUpdated,
48
+ RoutingSessionEnded,
49
+ SlotSet,
50
+ UserUttered,
51
+ ActionExecuted,
52
+ Event,
53
+ Restarted,
54
+ ActionReverted,
55
+ UserUtteranceReverted,
56
+ BotUttered,
57
+ ActiveLoop,
58
+ SessionStarted,
59
+ ActionExecutionRejected,
60
+ DefinePrevUserUtteredFeaturization,
61
+ FlowStarted,
62
+ )
63
+ from rasa.shared.core.flows import FlowsList
64
+ from rasa.shared.core.slots import AnySlot, Slot
65
+ from rasa.shared.nlu.constants import (
66
+ ENTITY_ATTRIBUTE_VALUE,
67
+ ENTITY_ATTRIBUTE_TYPE,
68
+ ENTITY_ATTRIBUTE_GROUP,
69
+ ENTITY_ATTRIBUTE_ROLE,
70
+ ACTION_TEXT,
71
+ ACTION_NAME,
72
+ ENTITIES,
73
+ METADATA_MODEL_ID,
74
+ )
75
+
76
+ if TYPE_CHECKING:
77
+ from rasa.shared.core.events import NLUPredictionData
78
+ from rasa.shared.core.training_data.structures import Story
79
+ from rasa.shared.core.training_data.story_writer.story_writer import StoryWriter
80
+ from rasa.dialogue_understanding.stack.dialogue_stack import DialogueStack
81
+
82
+ EventTypeAlias = TypeVar("EventTypeAlias", bound=Event)
83
+
84
+
85
+ @dataclasses.dataclass
86
+ class TrackerActiveLoop:
87
+ """Dataclass for `DialogueStateTracker.active_loop`."""
88
+
89
+ name: Optional[Text]
90
+ is_interrupted: bool
91
+ rejected: bool
92
+ trigger_message: Optional[Dict]
93
+
94
+
95
+ logger = logging.getLogger(__name__)
96
+
97
+ # same as State but with Dict[...] substituted with FrozenSet[Tuple[...]]
98
+ FrozenState = FrozenSet[Tuple[Text, FrozenSet[Tuple[Text, Tuple[Union[float, Text]]]]]]
99
+
100
+
101
+ class EventVerbosity(Enum):
102
+ """Filter on which events to include in tracker dumps."""
103
+
104
+ # no events will be included
105
+ NONE = 1
106
+
107
+ # all events, that contribute to the trackers state are included
108
+ # these are all you need to reconstruct the tracker state
109
+ APPLIED = 2
110
+
111
+ # include even more events, in this case everything that comes
112
+ # after the most recent restart event. this will also include
113
+ # utterances that got reverted and actions that got undone.
114
+ AFTER_RESTART = 3
115
+
116
+ # include every logged event
117
+ ALL = 4
118
+
119
+
120
+ class AnySlotDict(dict):
121
+ """A slot dictionary that pretends every slot exists, by creating slots on demand.
122
+
123
+ This only uses the generic slot type! This means certain functionality wont work,
124
+ e.g. properly featurizing the slot.
125
+ """
126
+
127
+ def __missing__(self, key: Text) -> Slot:
128
+ value = self[key] = AnySlot(key, mappings=[])
129
+ return value
130
+
131
+ def __contains__(self, key: Any) -> bool:
132
+ return True
133
+
134
+
135
+ class DialogueStateTracker:
136
+ """Maintains the state of a conversation.
137
+
138
+ The field max_event_history will only give you these last events,
139
+ it can be set in the tracker_store.
140
+ """
141
+
142
+ @classmethod
143
+ def from_dict(
144
+ cls,
145
+ sender_id: Text,
146
+ events_as_dict: List[Dict[Text, Any]],
147
+ slots: Optional[Iterable[Slot]] = None,
148
+ max_event_history: Optional[int] = None,
149
+ ) -> "DialogueStateTracker":
150
+ """Create a tracker from dump.
151
+
152
+ The dump should be an array of dumped events. When restoring
153
+ the tracker, these events will be replayed to recreate the state.
154
+ """
155
+ evts = events.deserialise_events(events_as_dict)
156
+
157
+ return cls.from_events(sender_id, evts, slots, max_event_history)
158
+
159
+ @classmethod
160
+ def from_events(
161
+ cls,
162
+ sender_id: Text,
163
+ evts: List[Event],
164
+ slots: Optional[Iterable[Slot]] = None,
165
+ max_event_history: Optional[int] = None,
166
+ sender_source: Optional[Text] = None,
167
+ domain: Optional[Domain] = None,
168
+ ) -> "DialogueStateTracker":
169
+ """Creates tracker from existing events.
170
+
171
+ Args:
172
+ sender_id: The ID of the conversation.
173
+ evts: Existing events which should be applied to the new tracker.
174
+ slots: Slots which can be set.
175
+ max_event_history: Maximum number of events which should be stored.
176
+ sender_source: File source of the messages.
177
+ domain: The current model domain.
178
+
179
+ Returns:
180
+ Instantiated tracker with its state updated according to the given
181
+ events.
182
+ """
183
+ tracker = cls(sender_id, slots, max_event_history, sender_source)
184
+
185
+ for e in evts:
186
+ tracker.update(e, domain)
187
+
188
+ return tracker
189
+
190
+ def __init__(
191
+ self,
192
+ sender_id: Text,
193
+ slots: Optional[Iterable[Slot]],
194
+ max_event_history: Optional[int] = None,
195
+ sender_source: Optional[Text] = None,
196
+ is_rule_tracker: bool = False,
197
+ ) -> None:
198
+ """Initialize the tracker.
199
+
200
+ A set of events can be stored externally, and we will run through all
201
+ of them to get the current state. The tracker will represent all the
202
+ information we captured while processing messages of the dialogue.
203
+ """
204
+ from rasa.dialogue_understanding.stack.dialogue_stack import DialogueStack
205
+
206
+ # maximum number of events to store
207
+ self._max_event_history = max_event_history
208
+ # list of previously seen events
209
+ self.events = self._create_events([])
210
+ # id of the source of the messages
211
+ self.sender_id = sender_id
212
+ # slots that can be filled in this domain
213
+ self.slots: Dict[str, Slot] = AnySlotDict()
214
+ if slots is not None:
215
+ self.slots = {slot.name: copy.copy(slot) for slot in slots}
216
+
217
+ self._underlying_stack = DialogueStack.empty()
218
+ # file source of the messages
219
+ self.sender_source = sender_source
220
+ # whether the tracker belongs to a rule-based data
221
+ self.is_rule_tracker = is_rule_tracker
222
+
223
+ ###
224
+ # current state of the tracker - MUST be re-creatable by processing
225
+ # all the events. This only defines the attributes, values are set in
226
+ # `reset()`
227
+ ###
228
+ # if tracker is paused, no actions should be taken
229
+ self._paused = False
230
+ # A deterministically scheduled action to be executed next
231
+ self.followup_action: Optional[Text] = ACTION_LISTEN_NAME
232
+ self.latest_action: Optional[Dict[Text, Text]] = None
233
+ # Stores the most recent message sent by the user
234
+ self.latest_message: Optional[UserUttered] = None
235
+ self.latest_bot_utterance: Optional[BotUttered] = None
236
+ self._reset()
237
+ self.active_loop: Optional[TrackerActiveLoop] = None
238
+
239
+ # Optional model_id to add to all events.
240
+ self.model_id: Optional[Text] = None
241
+ self.assistant_id: Optional[Text] = None
242
+
243
+ ###
244
+ # Public tracker interface
245
+ ###
246
+ def current_state(
247
+ self, event_verbosity: EventVerbosity = EventVerbosity.NONE
248
+ ) -> Dict[Text, Any]:
249
+ """Returns the current tracker state as an object."""
250
+ events = self._events_for_verbosity(event_verbosity)
251
+ events_as_dict = [e.as_dict() for e in events] if events is not None else None
252
+ latest_event_time = None
253
+ if len(self.events) > 0:
254
+ latest_event_time = self.events[-1].timestamp
255
+
256
+ return {
257
+ "sender_id": self.sender_id,
258
+ "slots": self.current_slot_values(),
259
+ "latest_message": self._latest_message_data(),
260
+ "latest_event_time": latest_event_time,
261
+ FOLLOWUP_ACTION: self.followup_action,
262
+ "paused": self.is_paused(),
263
+ "stack": self.stack.as_dict(),
264
+ "events": events_as_dict,
265
+ "latest_input_channel": self.get_latest_input_channel(),
266
+ ACTIVE_LOOP: (
267
+ dataclasses.asdict(self.active_loop) if self.active_loop else {}
268
+ ),
269
+ "latest_action": self.latest_action,
270
+ "latest_action_name": self.latest_action_name,
271
+ }
272
+
273
+ def _events_for_verbosity(
274
+ self, event_verbosity: EventVerbosity
275
+ ) -> Optional[List[Event]]:
276
+ if event_verbosity == EventVerbosity.ALL:
277
+ return list(self.events)
278
+ if event_verbosity == EventVerbosity.AFTER_RESTART:
279
+ return self.events_after_latest_restart()
280
+ if event_verbosity == EventVerbosity.APPLIED:
281
+ return self.applied_events()
282
+
283
+ return None
284
+
285
+ def _latest_message_data(self) -> Optional["NLUPredictionData"]:
286
+ if not self.latest_message:
287
+ return None
288
+
289
+ parse_data_with_nlu_state = self.latest_message.parse_data.copy()
290
+ # Combine entities predicted by NLU with entities predicted by policies so that
291
+ # users can access them together via `latest_message` (e.g. in custom actions)
292
+ parse_data_with_nlu_state[ENTITIES] = self.latest_message.entities # type: ignore[literal-required]
293
+
294
+ return parse_data_with_nlu_state
295
+
296
+ @staticmethod
297
+ def freeze_current_state(state: State) -> FrozenState:
298
+ """Convert State dict into a hashable format FrozenState.
299
+
300
+ Args:
301
+ state: The state which should be converted
302
+
303
+ Return:
304
+ hashable form of the state of type `FrozenState`
305
+ """
306
+ return frozenset(
307
+ {
308
+ key: frozenset(values.items())
309
+ if isinstance(values, Dict)
310
+ else frozenset(values)
311
+ for key, values in state.items()
312
+ }.items()
313
+ )
314
+
315
+ def past_states(
316
+ self,
317
+ domain: Domain,
318
+ omit_unset_slots: bool = False,
319
+ ignore_rule_only_turns: bool = False,
320
+ rule_only_data: Optional[Dict[Text, Any]] = None,
321
+ ) -> List[State]:
322
+ """Generates the past states of this tracker based on the history.
323
+
324
+ Args:
325
+ domain: The Domain.
326
+ omit_unset_slots: If `True` do not include the initial values of slots.
327
+ ignore_rule_only_turns: If True ignore dialogue turns that are present
328
+ only in rules.
329
+ rule_only_data: Slots and loops,
330
+ which only occur in rules but not in stories.
331
+
332
+ Returns:
333
+ A list of states
334
+ """
335
+ return domain.states_for_tracker_history(
336
+ self,
337
+ omit_unset_slots=omit_unset_slots,
338
+ ignore_rule_only_turns=ignore_rule_only_turns,
339
+ rule_only_data=rule_only_data,
340
+ )
341
+
342
+ def change_loop_to(self, loop_name: Optional[Text]) -> None:
343
+ """Set the currently active loop.
344
+
345
+ Args:
346
+ loop_name: The name of loop which should be marked as active.
347
+ """
348
+ if loop_name is not None:
349
+ self.active_loop = TrackerActiveLoop(
350
+ loop_name,
351
+ False,
352
+ False,
353
+ self.latest_message.parse_data if self.latest_message else None,
354
+ )
355
+ else:
356
+ self.active_loop = None
357
+
358
+ def interrupt_loop(self, is_interrupted: bool) -> None:
359
+ """Interrupt loop and mark that we entered an unhappy path in the conversation.
360
+
361
+ Args:
362
+ is_interrupted: `True` if the loop was run after an unhappy path.
363
+ """
364
+ if self.active_loop is not None:
365
+ self.active_loop.is_interrupted = is_interrupted
366
+
367
+ def reject_action(self, action_name: Text) -> None:
368
+ """Notify active loop that it was rejected."""
369
+ if self.active_loop is not None and action_name == self.active_loop_name:
370
+ self.active_loop.rejected = True
371
+
372
+ def set_latest_action(self, action: Dict[Text, Text]) -> None:
373
+ """Sets latest action name or text.
374
+
375
+ Resets loop validation and rejection parameters.
376
+
377
+ Args:
378
+ action: Serialized action event.
379
+ """
380
+ self.latest_action = action
381
+ if self.active_loop is not None and self.active_loop_name:
382
+ # reset form validation if some loop is active
383
+ self.active_loop.is_interrupted = False
384
+
385
+ if (
386
+ self.active_loop is not None
387
+ and action.get(ACTION_NAME) == self.active_loop_name
388
+ ):
389
+ # reset loop rejection if it was predicted again
390
+ self.active_loop.rejected = False
391
+
392
+ def current_slot_values(self) -> Dict[Text, Any]:
393
+ """Return the currently set values of the slots."""
394
+ return {key: slot.value for key, slot in self.slots.items()}
395
+
396
+ def get_slot(self, key: Text) -> Optional[Any]:
397
+ """Retrieves the value of a slot."""
398
+ if key in self.slots:
399
+ return self.slots[key].value
400
+ else:
401
+ logger.info(f"Tried to access non existent slot '{key}'")
402
+ return None
403
+
404
+ def create_stack_updated_events(
405
+ self, updated_stack: "DialogueStack"
406
+ ) -> List[Event]:
407
+ """Creates events to update the stack to the given one."""
408
+ patch = self._underlying_stack.create_stack_patch(updated_stack)
409
+
410
+ # if there is no patch, this is a no-op
411
+ if patch:
412
+ return [DialogueStackUpdated(patch)]
413
+ return []
414
+
415
+ def update_stack(self, updated_stack: "DialogueStack") -> None:
416
+ """Set's the updated stack on this tracker."""
417
+ for event in self.create_stack_updated_events(updated_stack):
418
+ self.update(event)
419
+
420
+ def apply_stack_update(self, update: str) -> None:
421
+ self._underlying_stack = self._underlying_stack.update_from_patch(update)
422
+
423
+ def previous_stack_states(self) -> Generator["DialogueStack", None, None]:
424
+ """Generates the previous stack states of this tracker."""
425
+ from rasa.dialogue_understanding.stack.dialogue_stack import DialogueStack
426
+
427
+ tracker = self.init_copy()
428
+ previous_stack = DialogueStack.empty()
429
+ yield previous_stack
430
+ for event in self.applied_events():
431
+ tracker.update(event)
432
+ stack = tracker.stack
433
+ if stack != previous_stack:
434
+ previous_stack = stack
435
+ yield stack
436
+
437
+ @property
438
+ def stack(self) -> "DialogueStack":
439
+ """Returns the current stack as a copy.
440
+
441
+ Important, modifying the returned stack does not modify the stack
442
+ stored on the tracker.
443
+ """
444
+ return self._underlying_stack.copy()
445
+
446
+ @property
447
+ def active_flow(self) -> Optional[Text]:
448
+ """Returns the name of the active flow."""
449
+ current_context = self.stack.current_context()
450
+ return current_context.get("flow_id")
451
+
452
+ @property
453
+ def has_coexistence_routing_slot(self) -> bool:
454
+ """Returns whether the coexistence routing slot is present."""
455
+ if self.slots:
456
+ return ROUTE_TO_CALM_SLOT in self.slots
457
+ return False
458
+
459
+ def has_bot_message_after_latest_user_message(self) -> bool:
460
+ """Checks if there is a bot message after the most recent user message.
461
+
462
+ Returns:
463
+ `True` if there is an action after the most recent user message.
464
+ """
465
+ for event in reversed(self.applied_events()):
466
+ if isinstance(event, BotUttered):
467
+ return True
468
+ elif isinstance(event, UserUttered):
469
+ return False
470
+ return False
471
+
472
+ def has_action_after_latest_user_message(self) -> bool:
473
+ """Check if there is an action after the most recent user message.
474
+
475
+ Returns:
476
+ `True` if there is an action after the most recent user message.
477
+ """
478
+ for event in reversed(self.applied_events()):
479
+ if isinstance(event, ActionExecuted):
480
+ return True
481
+ elif isinstance(event, UserUttered):
482
+ return False
483
+ return False
484
+
485
+ def get_latest_entity_values(
486
+ self,
487
+ entity_type: Text,
488
+ entity_role: Optional[Text] = None,
489
+ entity_group: Optional[Text] = None,
490
+ ) -> Iterator[Text]:
491
+ """Get entity values for latest message.
492
+
493
+ Returns entity values found for the passed entity type and
494
+ optional role and group in latest message.
495
+
496
+ If you are only interested in the first entity of a given type use
497
+ `next(tracker.get_latest_entity_values(`"`my_entity_name`"`), None)`.
498
+ If no entity is found `None` is the default result.
499
+
500
+ Args:
501
+ entity_type: the entity type of interest
502
+ entity_role: optional entity role of interest
503
+ entity_group: optional entity group of interest
504
+
505
+ Returns:
506
+ Entity values.
507
+ """
508
+ if self.latest_message is None:
509
+ return iter([])
510
+
511
+ return (
512
+ cast(Text, x[ENTITY_ATTRIBUTE_VALUE])
513
+ for x in self.latest_message.entities
514
+ if x.get(ENTITY_ATTRIBUTE_TYPE) == entity_type
515
+ and x.get(ENTITY_ATTRIBUTE_GROUP) == entity_group
516
+ and x.get(ENTITY_ATTRIBUTE_ROLE) == entity_role
517
+ )
518
+
519
+ def get_latest_input_channel(self) -> Optional[Text]:
520
+ """Get the name of the input_channel of the latest UserUttered event."""
521
+ for e in reversed(self.events):
522
+ if isinstance(e, UserUttered):
523
+ return e.input_channel
524
+ return None
525
+
526
+ def is_paused(self) -> bool:
527
+ """State whether the tracker is currently paused."""
528
+ return self._paused
529
+
530
+ def idx_after_latest_restart(self) -> int:
531
+ """Return the idx of the most recent restart in the list of events.
532
+
533
+ If the conversation has not been restarted, ``0`` is returned.
534
+ """
535
+ for i, event in enumerate(reversed(self.events)):
536
+ if isinstance(event, Restarted):
537
+ return len(self.events) - i
538
+
539
+ return 0
540
+
541
+ def events_after_latest_restart(self) -> List[Event]:
542
+ """Return a list of events after the most recent restart."""
543
+ return list(self.events)[self.idx_after_latest_restart() :]
544
+
545
+ def init_copy(self) -> "DialogueStateTracker":
546
+ """Creates a new state tracker with the same initial values."""
547
+ return DialogueStateTracker(
548
+ self.sender_id or DEFAULT_SENDER_ID,
549
+ self.slots.values(),
550
+ self._max_event_history,
551
+ is_rule_tracker=self.is_rule_tracker,
552
+ )
553
+
554
+ def generate_all_prior_trackers(
555
+ self,
556
+ ) -> Generator[Tuple["DialogueStateTracker", bool], None, None]:
557
+ """Returns a generator of the previous trackers of this tracker.
558
+
559
+ Returns:
560
+ The tuple with the tracker before each action,
561
+ and the boolean flag representing whether this action should be hidden
562
+ in the dialogue history created for ML-based policies.
563
+ """
564
+ tracker = self.init_copy()
565
+
566
+ for event in self.applied_events(True):
567
+ if isinstance(event, ActionExecuted):
568
+ yield tracker, event.hide_rule_turn
569
+
570
+ tracker.update(event)
571
+
572
+ yield tracker, False
573
+
574
+ def applied_events(self, featurization_for_policies: bool = False) -> List[Event]:
575
+ """Returns all actions that should be applied - w/o reverted events.
576
+
577
+ Returns:
578
+ The events applied to the tracker.
579
+ """
580
+ loop_names = [
581
+ event.name
582
+ for event in self.events
583
+ if isinstance(event, ActiveLoop) and event.name
584
+ ]
585
+
586
+ applied_events: List[Event] = []
587
+
588
+ for event in self.events:
589
+ if isinstance(event, (Restarted, SessionStarted)):
590
+ applied_events = []
591
+ elif isinstance(event, RoutingSessionEnded) and featurization_for_policies:
592
+ # remove all events but the set slots events for the slots that are
593
+ # shared for coexistence.
594
+ applied_events = [
595
+ e
596
+ for e in applied_events
597
+ if isinstance(e, SlotSet)
598
+ and (slot := self.slots.get(e.key)) is not None
599
+ and slot.shared_for_coexistence
600
+ ]
601
+ elif isinstance(event, ActionReverted):
602
+ self._undo_till_previous(ActionExecuted, applied_events)
603
+ elif isinstance(event, UserUtteranceReverted):
604
+ # Seeing a user uttered event automatically implies there was
605
+ # a listen event right before it, so we'll first rewind the
606
+ # user utterance, then get the action right before it (also removes
607
+ # the `action_listen` action right before it).
608
+ self._undo_till_previous(UserUttered, applied_events)
609
+ self._undo_till_previous(ActionExecuted, applied_events)
610
+ elif (
611
+ isinstance(event, ActionExecuted)
612
+ and event.action_name in loop_names
613
+ and not self._first_loop_execution_or_unhappy_path(
614
+ event.action_name, applied_events
615
+ )
616
+ ):
617
+ self._undo_till_previous_loop_execution(
618
+ event.action_name, applied_events
619
+ )
620
+ else:
621
+ applied_events.append(event)
622
+
623
+ return applied_events
624
+
625
+ @staticmethod
626
+ def _undo_till_previous(event_type: Type[Event], done_events: List[Event]) -> None:
627
+ """Removes events from `done_events`.
628
+
629
+ Removes events from `done_events` until the first occurrence `event_type`
630
+ is found which is also removed.
631
+ """
632
+ # list gets modified - hence we need to copy events!
633
+ for e in reversed(done_events[:]):
634
+ del done_events[-1]
635
+ if isinstance(e, event_type):
636
+ break
637
+
638
+ def _first_loop_execution_or_unhappy_path(
639
+ self, loop_action_name: Text, applied_events: List[Event]
640
+ ) -> bool:
641
+ next_action: Optional[Text] = None
642
+
643
+ for event in reversed(applied_events):
644
+ # Stop looking for a previous loop execution if there is a loop deactivation
645
+ # event because it means that the current loop is running for the first
646
+ # time and previous loop events belong to different loops.
647
+ if isinstance(event, ActiveLoop) and event.name is None:
648
+ return True
649
+
650
+ if self._is_within_unhappy_path(loop_action_name, event, next_action):
651
+ return True
652
+
653
+ if isinstance(event, ActionExecuted):
654
+ # We found a previous execution of the loop and we are not within an
655
+ # unhappy path.
656
+ if event.action_name == loop_action_name:
657
+ return False
658
+
659
+ # Remember the action as we need that to check whether we might be
660
+ # within an unhappy path.
661
+ next_action = event.action_name
662
+
663
+ return True
664
+
665
+ @staticmethod
666
+ def _is_within_unhappy_path(
667
+ loop_action_name: Text, event: Event, next_action_in_the_future: Optional[Text]
668
+ ) -> bool:
669
+ # When actual users are talking to the action has to return an
670
+ # `ActionExecutionRejected` in order to enter an unhappy path.
671
+ loop_was_rejected_previously = (
672
+ isinstance(event, ActionExecutionRejected)
673
+ and event.action_name == loop_action_name
674
+ )
675
+ # During the policy training there are no `ActionExecutionRejected` events
676
+ # which let us see whether we are within an unhappy path. Hence, we check if a
677
+ # different action was executed instead of the loop after last user utterance.
678
+ other_action_after_latest_user_utterance = (
679
+ isinstance(event, UserUttered)
680
+ and next_action_in_the_future is not None
681
+ and next_action_in_the_future != loop_action_name
682
+ )
683
+
684
+ return loop_was_rejected_previously or other_action_after_latest_user_utterance
685
+
686
+ @staticmethod
687
+ def _undo_till_previous_loop_execution(
688
+ loop_action_name: Text, done_events: List[Event]
689
+ ) -> None:
690
+ offset = 0
691
+ for e in reversed(done_events[:]):
692
+ if isinstance(e, ActionExecuted) and e.action_name == loop_action_name:
693
+ break
694
+
695
+ if isinstance(
696
+ e, (ActionExecuted, UserUttered, DefinePrevUserUtteredFeaturization)
697
+ ):
698
+ del done_events[-1 - offset]
699
+ else:
700
+ # Remember events which aren't unfeaturized to get the index right
701
+ offset += 1
702
+
703
+ def replay_events(self) -> None:
704
+ """Update the tracker based on a list of events."""
705
+ applied_events = self.applied_events()
706
+ for event in applied_events:
707
+ event.apply_to(self)
708
+
709
+ def recreate_from_dialogue(self, dialogue: Dialogue) -> None:
710
+ """Use a serialised `Dialogue` to update the trackers state.
711
+
712
+ This uses the state as is persisted in a ``TrackerStore``. If the
713
+ tracker is blank before calling this method, the final state will be
714
+ identical to the tracker from which the dialogue was created.
715
+ """
716
+ if not isinstance(dialogue, Dialogue):
717
+ raise ValueError(
718
+ f"story {dialogue} is not of type Dialogue. "
719
+ f"Have you deserialized it?"
720
+ )
721
+
722
+ self._reset()
723
+ self.events.extend(dialogue.events)
724
+ self.replay_events()
725
+
726
+ def copy(self) -> "DialogueStateTracker":
727
+ """Creates a duplicate of this tracker."""
728
+ return copy.deepcopy(self)
729
+
730
+ def travel_back_in_time(self, target_time: float) -> "DialogueStateTracker":
731
+ """Creates a new tracker with a state at a specific timestamp.
732
+
733
+ A new tracker will be created and all events previous to the
734
+ passed time stamp will be replayed. Events that occur exactly
735
+ at the target time will be included.
736
+ """
737
+ tracker = self.init_copy()
738
+
739
+ for event in self.events:
740
+ if event.timestamp <= target_time:
741
+ tracker.update(event)
742
+ else:
743
+ break
744
+
745
+ return tracker # yields the final state
746
+
747
+ def as_dialogue(self) -> Dialogue:
748
+ """Return a ``Dialogue`` object containing all of the turns.
749
+
750
+ This can be serialised and later used to recover the state
751
+ of this tracker exactly.
752
+ """
753
+ return Dialogue(self.sender_id, list(self.events))
754
+
755
+ def update(self, event: Event, domain: Optional[Domain] = None) -> None:
756
+ """Modify the state of the tracker according to an ``Event``."""
757
+ if not isinstance(event, Event): # pragma: no cover
758
+ raise ValueError("event to log must be an instance of a subclass of Event.")
759
+
760
+ if self.model_id and METADATA_MODEL_ID not in event.metadata:
761
+ event.metadata = {**event.metadata, METADATA_MODEL_ID: self.model_id}
762
+
763
+ if self.assistant_id and ASSISTANT_ID_KEY not in event.metadata:
764
+ event.metadata = {**event.metadata, ASSISTANT_ID_KEY: self.assistant_id}
765
+
766
+ self.events.append(event)
767
+ event.apply_to(self)
768
+
769
+ def update_with_events(
770
+ self,
771
+ new_events: List[Event],
772
+ # TODO: remove domain argument - breaking change.
773
+ domain: Optional[Domain] = None,
774
+ override_timestamp: bool = True,
775
+ ) -> None:
776
+ """Adds multiple events to the tracker.
777
+
778
+ Args:
779
+ new_events: Events to apply.
780
+ domain: The current model's domain. Not needed anymore.
781
+ kept for backwards compatibility.
782
+ override_timestamp: If `True` refresh all timestamps of the events. As the
783
+ events are usually created at some earlier point, this makes sure that
784
+ all new events come after any current tracker events.
785
+ """
786
+ for e in new_events:
787
+ if override_timestamp:
788
+ e.timestamp = time.time()
789
+ self.update(e, domain)
790
+
791
+ def as_story(self, include_source: bool = False) -> "Story":
792
+ """Dump the tracker as a story in the Rasa Core story format.
793
+
794
+ Returns the dumped tracker as a string.
795
+ """
796
+ from rasa.shared.core.training_data.structures import Story
797
+
798
+ story_name = (
799
+ f"{self.sender_id} ({self.sender_source})"
800
+ if include_source
801
+ else self.sender_id
802
+ )
803
+ return Story.from_events(list(self.events), story_name)
804
+
805
+ def export_stories(
806
+ self,
807
+ writer: "StoryWriter",
808
+ e2e: bool = False,
809
+ include_source: bool = False,
810
+ should_append_stories: bool = False,
811
+ ) -> Text:
812
+ """Dump the tracker as a story in the Rasa Core story format.
813
+
814
+ Returns:
815
+ The dumped tracker as a string.
816
+ """
817
+ story = self.as_story(include_source)
818
+ return writer.dumps(
819
+ story.story_steps, is_appendable=should_append_stories, is_test_story=e2e
820
+ )
821
+
822
+ def export_stories_to_file(self, export_path: Text = "debug_stories.yml") -> None:
823
+ """Dump the tracker as a story to a file."""
824
+ from rasa.shared.core.training_data.story_writer.yaml_story_writer import (
825
+ YAMLStoryWriter,
826
+ )
827
+
828
+ append = os.path.exists(export_path)
829
+
830
+ rasa.shared.utils.io.write_text_file(
831
+ self.export_stories(YAMLStoryWriter(), should_append_stories=append) + "\n",
832
+ export_path,
833
+ append=append,
834
+ )
835
+
836
+ def get_last_event_for(
837
+ self,
838
+ event_type: Union[Type["EventTypeAlias"], Tuple[Type["EventTypeAlias"], ...]],
839
+ action_names_to_exclude: Optional[List[Text]] = None,
840
+ skip: int = 0,
841
+ event_verbosity: EventVerbosity = EventVerbosity.APPLIED,
842
+ ) -> Optional["EventTypeAlias"]:
843
+ """Gets the last event of a given type which was actually applied.
844
+
845
+ Args:
846
+ event_type: The type of event you want to find.
847
+ action_names_to_exclude: Events of type `ActionExecuted` which
848
+ should be excluded from the results. Can be used to skip
849
+ `action_listen` events.
850
+ skip: Skips n possible results before return an event.
851
+ event_verbosity: Which `EventVerbosity` should be used to search for events.
852
+
853
+ Returns:
854
+ event which matched the query or `None` if no event matched.
855
+ """
856
+ to_exclude = action_names_to_exclude or []
857
+
858
+ def filter_function(e: Event) -> bool:
859
+ has_instance = isinstance(e, event_type)
860
+ excluded = isinstance(e, ActionExecuted) and e.action_name in to_exclude
861
+ return has_instance and not excluded
862
+
863
+ filtered = filter(
864
+ filter_function, reversed(self._events_for_verbosity(event_verbosity) or [])
865
+ )
866
+
867
+ for i in range(skip):
868
+ next(filtered, None)
869
+
870
+ return next(filtered, None)
871
+
872
+ def last_executed_action_has(self, name: Text, skip: int = 0) -> bool:
873
+ """Returns whether last `ActionExecuted` event had a specific name.
874
+
875
+ Args:
876
+ name: Name of the event which should be matched.
877
+ skip: Skips n possible results in between.
878
+
879
+ Returns:
880
+ `True` if last executed action had name `name`, otherwise `False`.
881
+ """
882
+ last: Optional[ActionExecuted] = self.get_last_event_for(
883
+ ActionExecuted, action_names_to_exclude=[ACTION_LISTEN_NAME], skip=skip
884
+ )
885
+ return last is not None and last.action_name == name
886
+
887
+ ###
888
+ # Internal methods for the modification of the trackers state. Should
889
+ # only be called by events, not directly. Rather update the tracker
890
+ # with an event that in its ``apply_to`` method modifies the tracker.
891
+ ###
892
+ def _reset(self, is_coexistence_reset: bool = False) -> None:
893
+ """Reset tracker to initial state - doesn't delete events though!."""
894
+ from rasa.dialogue_understanding.stack.dialogue_stack import DialogueStack
895
+
896
+ self._reset_slots(is_coexistence_reset)
897
+ self._paused = False
898
+ self.latest_action = {}
899
+ self.latest_message = UserUttered.empty()
900
+ self.latest_bot_utterance = BotUttered.empty()
901
+ self.followup_action = ACTION_LISTEN_NAME
902
+ self.active_loop = None
903
+ self._underlying_stack = DialogueStack.empty()
904
+
905
+ def _reset_slots(self, is_coexistence_reset: bool = False) -> None:
906
+ """Set all the slots to their initial value."""
907
+ for slot in self.slots.values():
908
+ # skip slots with shared_for_coexistence during this reset
909
+ if is_coexistence_reset and slot.shared_for_coexistence:
910
+ continue
911
+ slot.reset()
912
+
913
+ def _set_slot(self, key: Text, value: Any) -> None:
914
+ """Sets the value of a slot if that slot exists."""
915
+ if key in self.slots:
916
+ slot = self.slots[key]
917
+ slot.value = value
918
+ else:
919
+ logger.error(
920
+ f"Tried to set non existent slot '{key}'. Make sure you "
921
+ f"added all your slots to your domain file."
922
+ )
923
+
924
+ def _create_events(self, evts: List[Event]) -> Deque[Event]:
925
+ if evts and not isinstance(evts[0], Event): # pragma: no cover
926
+ raise ValueError("events, if given, must be a list of events")
927
+ return deque(evts, self._max_event_history)
928
+
929
+ def __eq__(self, other: Any) -> bool:
930
+ if isinstance(self, type(other)):
931
+ return other.events == self.events and self.sender_id == other.sender_id
932
+ else:
933
+ return False
934
+
935
+ def __ne__(self, other: Any) -> bool:
936
+ return not self.__eq__(other)
937
+
938
+ def __repr__(self) -> Text:
939
+ """Returns event as string for debugging."""
940
+ return f"DialogueStateTracker(sender_id: {self.sender_id})"
941
+
942
+ def __str__(self) -> Text:
943
+ """Returns event as human-readable string."""
944
+ return f"{self.__class__.__name__}({self.sender_id})"
945
+
946
+ def trigger_followup_action(self, action: Text) -> None:
947
+ """Triggers another action following the execution of the current."""
948
+ self.followup_action = action
949
+
950
+ def clear_followup_action(self) -> None:
951
+ """Clears follow up action when it was executed."""
952
+ self.followup_action = None
953
+
954
+ @property
955
+ def active_loop_name(self) -> Optional[Text]:
956
+ """Get the name of the currently active loop.
957
+
958
+ Returns: `None` if no active loop or the name of the currently active loop.
959
+ """
960
+ if not self.active_loop or self.active_loop.name == SHOULD_NOT_BE_SET:
961
+ return None
962
+
963
+ return self.active_loop.name
964
+
965
+ @property
966
+ def latest_action_name(self) -> Optional[Text]:
967
+ """Get the name of the previously executed action or text of e2e action.
968
+
969
+ Returns: name of the previously executed action or text of e2e action
970
+ """
971
+ if self.latest_action is None:
972
+ return None
973
+
974
+ return self.latest_action.get(ACTION_NAME) or self.latest_action.get(
975
+ ACTION_TEXT
976
+ )
977
+
978
+ @property
979
+ def is_active_loop_rejected(self) -> bool:
980
+ """Return True if there is an active loop and it's rejected."""
981
+ return self.active_loop is not None and self.active_loop.rejected
982
+
983
+ @property
984
+ def is_active_loop_interrupted(self) -> bool:
985
+ """Return True if there is an active loop and it's interrupted."""
986
+ return self.active_loop is not None and self.active_loop.is_interrupted
987
+
988
+ def fingerprint(self) -> Text:
989
+ """Returns a unique hash for the tracker which is stable across python runs.
990
+
991
+ Returns:
992
+ fingerprint of the tracker
993
+ """
994
+ data: Dict[Text, Any] = {"sender_id": self.sender_id}
995
+
996
+ if self.slots:
997
+ data.update(self.slots)
998
+
999
+ if self.events:
1000
+ data["events"] = list(self.events)
1001
+
1002
+ return rasa.shared.utils.io.get_dictionary_fingerprint(data)
1003
+
1004
+ def get_previously_started_flows(
1005
+ self,
1006
+ flows: FlowsList,
1007
+ max_turns: Optional[int] = 20,
1008
+ ) -> FlowsList:
1009
+ """Retrieves a list of previously started flows.
1010
+
1011
+ Returned flows have been started in the past within a given
1012
+ number of conversation turns.
1013
+
1014
+ Args:
1015
+ flows: list of flows to check against for started flows.
1016
+ max_turns: the maximum number of turns to include in the transcript.
1017
+
1018
+ Returns:
1019
+ List of flows that have been started within the specified number of turns
1020
+ """
1021
+ previously_started_flows = dict()
1022
+ turn_counter = 0
1023
+
1024
+ # cycle through events in reverse order (newest events are appended at the end)
1025
+ for event in reversed(self.events):
1026
+ # check for FlowStarted event and append flow if it's in the flows lists
1027
+ if (
1028
+ isinstance(event, FlowStarted)
1029
+ and event.flow_id not in previously_started_flows
1030
+ and (flow := flows.flow_by_id(event.flow_id))
1031
+ ):
1032
+ previously_started_flows[event.flow_id] = flow
1033
+
1034
+ # count turns only for user or bot utterances
1035
+ if isinstance(event, (UserUttered, BotUttered)):
1036
+ turn_counter += 1
1037
+ if max_turns is not None and turn_counter > max_turns:
1038
+ break
1039
+
1040
+ return FlowsList(underlying_flows=list(previously_started_flows.values()))
1041
+
1042
+ def get_startable_flows(self, flows: FlowsList) -> FlowsList:
1043
+ """Retrieves a list of flows that can be started.
1044
+
1045
+ Returned flows are startable given the current
1046
+ state (context and slot values) of the tracker.
1047
+
1048
+ Args:
1049
+ flows: list of flows to check against for startable flows.
1050
+
1051
+ Returns:
1052
+ List of flows that are startable within the current state of the tracker
1053
+ """
1054
+ # get the current context and slot values for the flow guard check
1055
+ context = self.stack.current_context()
1056
+ slots = self.slots
1057
+ return flows.get_startable_flows(context, slots)
1058
+
1059
+ @property
1060
+ def has_active_user_flow(self) -> bool:
1061
+ from rasa.dialogue_understanding.stack.utils import top_user_flow_frame
1062
+
1063
+ top_relevant_frame = top_user_flow_frame(self.stack)
1064
+ return bool(top_relevant_frame and top_relevant_frame.flow_id)
1065
+
1066
+ def get_active_flows(self, flows: FlowsList) -> FlowsList:
1067
+ """Retrieve a list of all currently active flows.
1068
+
1069
+ Args:
1070
+ flows: list of flows to check against for active flows.
1071
+
1072
+ Returns:
1073
+ List of flows that are active within the current state of the tracker
1074
+ """
1075
+ from rasa.dialogue_understanding.stack.frames import UserFlowStackFrame
1076
+ from rasa.dialogue_understanding.stack.frames.flow_stack_frame import (
1077
+ FlowStackFrameType,
1078
+ )
1079
+
1080
+ active_flows = []
1081
+ for frame in reversed(self.stack.frames):
1082
+ # The stack just contains the current active frames and we are just
1083
+ # interested in the user flow stack frames.
1084
+ if isinstance(frame, UserFlowStackFrame):
1085
+ active_flows.append(frame.flow(flows))
1086
+ if frame.frame_type != FlowStackFrameType.CALL:
1087
+ # Iterate unitl we reach a frame that is not a call frame.
1088
+ break
1089
+
1090
+ return FlowsList(active_flows)
1091
+
1092
+
1093
+ class TrackerEventDiffEngine:
1094
+ """Computes event difference of two trackers."""
1095
+
1096
+ @staticmethod
1097
+ def event_difference(
1098
+ original: DialogueStateTracker, tracker: DialogueStateTracker
1099
+ ) -> List[Event]:
1100
+ """Find all events in the tracker not present in the original tracker.
1101
+
1102
+ Args:
1103
+ original: Original tracker to compare against.
1104
+ tracker: Tracker containing events from the current conversation session.
1105
+
1106
+ Returns:
1107
+ List of events from the new tracker which are not present
1108
+ in the original tracker.
1109
+ """
1110
+ offset = len(original.events) if original else 0
1111
+ events = tracker.events
1112
+ return list(itertools.islice(events, offset, len(events)))
1113
+
1114
+
1115
+ def get_active_loop_name(
1116
+ state: State,
1117
+ ) -> Optional[Text]:
1118
+ """Get the name of current active loop.
1119
+
1120
+ Args:
1121
+ state: The state from which the name of active loop should be extracted
1122
+
1123
+ Return:
1124
+ the name of active loop or None
1125
+ """
1126
+ if (
1127
+ not state.get(ACTIVE_LOOP)
1128
+ or state[ACTIVE_LOOP].get(LOOP_NAME) == SHOULD_NOT_BE_SET
1129
+ ):
1130
+ return None
1131
+
1132
+ # FIXME: better type annotation for `State` would require
1133
+ # a larger refactoring (e.g. switch to dataclass)
1134
+ return cast(Optional[Text], state[ACTIVE_LOOP].get(LOOP_NAME))
1135
+
1136
+
1137
+ def is_prev_action_listen_in_state(state: State) -> bool:
1138
+ """Check if action_listen is the previous executed action.
1139
+
1140
+ Args:
1141
+ state: The state for which the check should be performed
1142
+
1143
+ Return:
1144
+ boolean value indicating whether action_listen is previous action
1145
+ """
1146
+ prev_action_name = state.get(PREVIOUS_ACTION, {}).get(ACTION_NAME)
1147
+ return prev_action_name == ACTION_LISTEN_NAME
1148
+
1149
+
1150
+ def get_trackers_for_conversation_sessions(
1151
+ tracker: DialogueStateTracker,
1152
+ ) -> List[DialogueStateTracker]:
1153
+ """Generate trackers for `tracker` that are split by conversation sessions.
1154
+
1155
+ Args:
1156
+ tracker: Instance of `DialogueStateTracker` to split.
1157
+
1158
+ Returns:
1159
+ The trackers split by conversation sessions.
1160
+ """
1161
+ split_conversations = events.split_events(
1162
+ tracker.events,
1163
+ ActionExecuted,
1164
+ {"action_name": ACTION_SESSION_START_NAME},
1165
+ include_splitting_event=True,
1166
+ )
1167
+
1168
+ return [
1169
+ DialogueStateTracker.from_events(
1170
+ tracker.sender_id,
1171
+ evts,
1172
+ tracker.slots.values(),
1173
+ sender_source=tracker.sender_source,
1174
+ max_event_history=tracker._max_event_history,
1175
+ )
1176
+ for evts in split_conversations
1177
+ ]