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,467 @@
1
+ from abc import ABC, abstractmethod
2
+ from functools import lru_cache
3
+ from typing import Dict, Any, List, Optional, Tuple, Union, Text
4
+
5
+ import structlog
6
+ from jinja2 import Template
7
+
8
+ import rasa.shared.utils.io
9
+ from rasa.dialogue_understanding.commands import (
10
+ Command,
11
+ StartFlowCommand,
12
+ )
13
+ from rasa.dialogue_understanding.generator import CommandGenerator
14
+ from rasa.dialogue_understanding.generator.constants import (
15
+ DEFAULT_LLM_CONFIG,
16
+ LLM_CONFIG_KEY,
17
+ FLOW_RETRIEVAL_KEY,
18
+ FLOW_RETRIEVAL_ACTIVE_KEY,
19
+ )
20
+ from rasa.dialogue_understanding.generator.flow_retrieval import FlowRetrieval
21
+ from rasa.engine.graph import GraphComponent, ExecutionContext
22
+ from rasa.engine.recipes.default_recipe import DefaultV1Recipe
23
+ from rasa.engine.storage.resource import Resource
24
+ from rasa.engine.storage.storage import ModelStorage
25
+ from rasa.shared.core.domain import Domain
26
+ from rasa.shared.core.flows import FlowStep, Flow, FlowsList
27
+ from rasa.shared.core.flows.steps.collect import CollectInformationFlowStep
28
+ from rasa.shared.core.trackers import DialogueStateTracker
29
+ from rasa.shared.exceptions import FileIOException
30
+ from rasa.shared.exceptions import ProviderClientAPIException
31
+ from rasa.shared.nlu.constants import FLOWS_IN_PROMPT
32
+ from rasa.shared.nlu.training_data.message import Message
33
+ from rasa.shared.nlu.training_data.training_data import TrainingData
34
+ from rasa.shared.utils.llm import (
35
+ llm_factory,
36
+ allowed_values_for_slot,
37
+ )
38
+ from rasa.utils.log_utils import log_llm
39
+
40
+ structlogger = structlog.get_logger()
41
+
42
+
43
+ @DefaultV1Recipe.register(
44
+ [
45
+ DefaultV1Recipe.ComponentType.COMMAND_GENERATOR,
46
+ ],
47
+ is_trainable=True,
48
+ )
49
+ class LLMBasedCommandGenerator(GraphComponent, CommandGenerator, ABC):
50
+ """An abstract class defining interface and common functionality
51
+ of an LLM-based command generators.
52
+ """
53
+
54
+ def __init__(
55
+ self,
56
+ config: Dict[str, Any],
57
+ model_storage: ModelStorage,
58
+ resource: Resource,
59
+ **kwargs: Any,
60
+ ) -> None:
61
+ super().__init__(config)
62
+ self.config = {**self.get_default_config(), **config}
63
+ self._model_storage = model_storage
64
+ self._resource = resource
65
+ self.flow_retrieval: Optional[FlowRetrieval]
66
+
67
+ if self.enabled_flow_retrieval:
68
+ self.flow_retrieval = FlowRetrieval(
69
+ self.config[FLOW_RETRIEVAL_KEY], model_storage, resource
70
+ )
71
+ structlogger.info("llm_based_command_generator.flow_retrieval.enabled")
72
+ else:
73
+ self.flow_retrieval = None
74
+ structlogger.warn(
75
+ "llm_based_command_generator.flow_retrieval.disabled",
76
+ event_info=(
77
+ "Disabling flow retrieval can cause issues when there are a "
78
+ "large number of flows to be included in the prompt. For more"
79
+ "information see:\n"
80
+ "https://rasa.com/docs/rasa-pro/concepts/dialogue-understanding#how-the-llmcommandgenerator-works"
81
+ ),
82
+ )
83
+
84
+ ### Abstract methods
85
+ @staticmethod
86
+ @abstractmethod
87
+ def get_default_config() -> Dict[str, Any]:
88
+ """The component's default config."""
89
+ pass
90
+
91
+ @classmethod
92
+ @abstractmethod
93
+ def load(
94
+ cls: Any,
95
+ config: Dict[str, Any],
96
+ model_storage: ModelStorage,
97
+ resource: Resource,
98
+ execution_context: ExecutionContext,
99
+ **kwargs: Any,
100
+ ) -> "LLMBasedCommandGenerator":
101
+ """Loads trained component (see parent class for full docstring)."""
102
+ pass
103
+
104
+ @abstractmethod
105
+ def persist(self) -> None:
106
+ """Persist the component to disk for future loading."""
107
+ pass
108
+
109
+ @abstractmethod
110
+ async def predict_commands(
111
+ self,
112
+ message: Message,
113
+ flows: FlowsList,
114
+ tracker: Optional[DialogueStateTracker] = None,
115
+ **kwargs: Any,
116
+ ) -> List[Command]:
117
+ """Predict commands using the LLM.
118
+
119
+ Args:
120
+ message: The message from the user.
121
+ flows: The flows available to the user.
122
+ tracker: The tracker containing the current state of the conversation.
123
+ **kwargs: Keyword arguments for forward compatibility.
124
+
125
+ Returns:
126
+ The commands generated by the llm.
127
+ """
128
+ pass
129
+
130
+ @abstractmethod
131
+ def parse_commands(
132
+ cls, actions: Optional[str], tracker: DialogueStateTracker, flows: FlowsList
133
+ ) -> List[Command]:
134
+ """Parse the actions returned by the llm into intent and entities.
135
+
136
+ Args:
137
+ actions: The actions returned by the llm.
138
+ tracker: The tracker containing the current state of the conversation.
139
+ flows: the list of flows
140
+
141
+ Returns:
142
+ The parsed commands.
143
+ """
144
+ pass
145
+
146
+ @classmethod
147
+ @abstractmethod
148
+ def fingerprint_addon(cls: Any, config: Dict[str, Any]) -> Optional[str]:
149
+ """Add a fingerprint of the knowledge base for the graph."""
150
+ pass
151
+
152
+ ### Shared implementations of GraphComponent parent
153
+ @classmethod
154
+ def create(
155
+ cls,
156
+ config: Dict[str, Any],
157
+ model_storage: ModelStorage,
158
+ resource: Resource,
159
+ execution_context: ExecutionContext,
160
+ ) -> "LLMBasedCommandGenerator":
161
+ """Creates a new untrained component (see parent class for full docstring)."""
162
+ return cls(config, model_storage, resource)
163
+
164
+ def train(
165
+ self, training_data: TrainingData, flows: FlowsList, domain: Domain
166
+ ) -> Resource:
167
+ """Train the llm based command generator. Stores all flows into a vector
168
+ store.
169
+ """
170
+ # flow retrieval is populated with only user-defined flows
171
+ try:
172
+ if self.flow_retrieval is not None and not flows.is_empty():
173
+ self.flow_retrieval.populate(flows.user_flows, domain)
174
+ except Exception as e:
175
+ structlogger.error(
176
+ "llm_based_command_generator.train.failed",
177
+ event_info=("Flow retrieval store isinaccessible."),
178
+ error=e,
179
+ )
180
+ raise
181
+ self.persist()
182
+ return self._resource
183
+
184
+ ### Helper methods
185
+ @property
186
+ def enabled_flow_retrieval(self) -> bool:
187
+ return self.config[FLOW_RETRIEVAL_KEY].get(FLOW_RETRIEVAL_ACTIVE_KEY, True)
188
+
189
+ @lru_cache
190
+ def compile_template(self, template: str) -> Template:
191
+ """Compile the prompt template.
192
+
193
+ Compiling the template is an expensive operation,
194
+ so we cache the result.
195
+ """
196
+ return Template(template)
197
+
198
+ @classmethod
199
+ def load_prompt_template_from_model_storage(
200
+ cls,
201
+ model_storage: ModelStorage,
202
+ resource: Resource,
203
+ prompt_template_file_name: str,
204
+ ) -> Optional[Text]:
205
+ try:
206
+ with model_storage.read_from(resource) as path:
207
+ return rasa.shared.utils.io.read_file(path / prompt_template_file_name)
208
+ except (FileNotFoundError, FileIOException) as e:
209
+ structlogger.warning(
210
+ "llm_based_command_generator.load_prompt_template.failed",
211
+ error=e,
212
+ resource=resource.name,
213
+ )
214
+ return None
215
+
216
+ @classmethod
217
+ def load_flow_retrival(
218
+ cls, config: Dict[Text, Any], model_storage: ModelStorage, resource: Resource
219
+ ) -> Optional[FlowRetrieval]:
220
+ """Load the FlowRetrieval component if it is enabled in the configuration."""
221
+ enable_flow_retrieval = config.get(FLOW_RETRIEVAL_KEY, {}).get(
222
+ FLOW_RETRIEVAL_ACTIVE_KEY, True
223
+ )
224
+ if enable_flow_retrieval:
225
+ return FlowRetrieval.load(
226
+ config=config.get(FLOW_RETRIEVAL_KEY),
227
+ model_storage=model_storage,
228
+ resource=resource,
229
+ )
230
+ return None
231
+
232
+ async def filter_flows(
233
+ self,
234
+ message: Message,
235
+ flows: FlowsList,
236
+ tracker: Optional[DialogueStateTracker] = None,
237
+ ) -> FlowsList:
238
+ """Filters the available flows.
239
+
240
+ Args:
241
+ message: The message from the user.
242
+ flows: The flows available to the user.
243
+ tracker: The tracker containing the current state of the conversation.
244
+
245
+ Returns:
246
+ Filtered list of flows.
247
+ """
248
+ # If the flow retrieval is disabled, use the all the provided flows.
249
+ filtered_flows = (
250
+ await self.flow_retrieval.filter_flows(tracker, message, flows)
251
+ if self.flow_retrieval is not None
252
+ else flows
253
+ )
254
+ # Filter flows based on current context (tracker and message)
255
+ # to identify which flows LLM can potentially start.
256
+ if tracker:
257
+ filtered_flows = tracker.get_startable_flows(filtered_flows)
258
+ else:
259
+ filtered_flows = filtered_flows
260
+
261
+ # add the filtered flows to the message for evaluation purposes
262
+ message.set(
263
+ FLOWS_IN_PROMPT, list(filtered_flows.user_flow_ids), add_to_output=True
264
+ )
265
+ log_llm(
266
+ logger=structlogger,
267
+ log_module="LLMBasedCommandGenerator",
268
+ log_event="llm_based_command_generator.predict_commands.filtered_flows",
269
+ message=message.data,
270
+ enabled_flow_retrieval=self.flow_retrieval is not None,
271
+ relevant_flows=list(filtered_flows.user_flow_ids),
272
+ )
273
+ return filtered_flows
274
+
275
+ async def invoke_llm(self, prompt: Text) -> Optional[Text]:
276
+ """Use LLM to generate a response.
277
+
278
+ Args:
279
+ prompt: The prompt to send to the LLM.
280
+
281
+ Returns:
282
+ The generated text.
283
+
284
+ Raises:
285
+ ProviderClientAPIException if an error during API call.
286
+ """
287
+ llm = llm_factory(self.config.get(LLM_CONFIG_KEY), DEFAULT_LLM_CONFIG)
288
+ try:
289
+ return await llm.apredict(prompt)
290
+ except Exception as e:
291
+ # unfortunately, langchain does not wrap LLM exceptions which means
292
+ # we have to catch all exceptions here
293
+ structlogger.error("llm_based_command_generator.llm.error", error=e)
294
+ raise ProviderClientAPIException(
295
+ message="LLM call exception", original_exception=e
296
+ )
297
+
298
+ @staticmethod
299
+ def start_flow_by_name(flow_name: str, flows: FlowsList) -> List[Command]:
300
+ """Start a flow by name.
301
+
302
+ If the flow does not exist, no command is returned.
303
+ """
304
+ if flow_name in flows.user_flow_ids:
305
+ return [StartFlowCommand(flow=flow_name)]
306
+ else:
307
+ structlogger.debug(
308
+ "llm_command_generator.flow.start_invalid_flow_id", flow=flow_name
309
+ )
310
+ return []
311
+
312
+ @staticmethod
313
+ def is_none_value(value: str) -> bool:
314
+ """Check if the value is a none value."""
315
+ return value in {
316
+ "[missing information]",
317
+ "[missing]",
318
+ "None",
319
+ "undefined",
320
+ "null",
321
+ }
322
+
323
+ @staticmethod
324
+ def clean_extracted_value(value: str) -> str:
325
+ """Clean up the extracted value from the llm."""
326
+ # replace any combination of single quotes, double quotes, and spaces
327
+ # from the beginning and end of the string
328
+ return value.strip("'\" ")
329
+
330
+ @classmethod
331
+ def get_nullable_slot_value(cls, slot_value: str) -> Union[str, None]:
332
+ """Get the slot value or None if the value is a none value.
333
+
334
+ Args:
335
+ slot_value: the value to coerce
336
+
337
+ Returns:
338
+ The slot value or None if the value is a none value.
339
+ """
340
+ return slot_value if not cls.is_none_value(slot_value) else None
341
+
342
+ def prepare_flows_for_template(
343
+ self, flows: FlowsList, tracker: DialogueStateTracker
344
+ ) -> List[Dict[str, Any]]:
345
+ """Format data on available flows for insertion into the prompt template.
346
+
347
+ Args:
348
+ flows: The flows available to the user.
349
+ tracker: The tracker containing the current state of the conversation.
350
+
351
+ Returns:
352
+ The inputs for the prompt template.
353
+ """
354
+ result = []
355
+ for flow in flows.user_flows:
356
+ slots_with_info = [
357
+ {
358
+ "name": q.collect,
359
+ "description": q.description,
360
+ "allowed_values": allowed_values_for_slot(tracker.slots[q.collect]),
361
+ }
362
+ for q in flow.get_collect_steps()
363
+ if self.is_extractable(q, tracker)
364
+ ]
365
+ result.append(
366
+ {
367
+ "name": flow.id,
368
+ "description": flow.description,
369
+ "slots": slots_with_info,
370
+ }
371
+ )
372
+ return result
373
+
374
+ @staticmethod
375
+ def is_extractable(
376
+ collect_step: CollectInformationFlowStep,
377
+ tracker: DialogueStateTracker,
378
+ current_step: Optional[FlowStep] = None,
379
+ ) -> bool:
380
+ """Check if the `collect` can be filled.
381
+
382
+ A collect slot can only be filled if the slot exist
383
+ and either the collect has been asked already or the
384
+ slot has been filled already.
385
+
386
+ Args:
387
+ collect_step: The collect_information step.
388
+ tracker: The tracker containing the current state of the conversation.
389
+ current_step: The current step in the flow.
390
+
391
+ Returns:
392
+ `True` if the slot can be filled, `False` otherwise.
393
+ """
394
+ slot = tracker.slots.get(collect_step.collect)
395
+ if slot is None:
396
+ return False
397
+
398
+ return (
399
+ # we can fill because this is a slot that can be filled ahead of time
400
+ not collect_step.ask_before_filling
401
+ # we can fill because the slot has been filled already
402
+ or slot.has_been_set
403
+ # we can fill because the is currently getting asked
404
+ or (
405
+ current_step is not None
406
+ and isinstance(current_step, CollectInformationFlowStep)
407
+ and current_step.collect == collect_step.collect
408
+ )
409
+ )
410
+
411
+ @staticmethod
412
+ def get_slot_value(tracker: DialogueStateTracker, slot_name: str) -> str:
413
+ """Get the slot value from the tracker.
414
+
415
+ Args:
416
+ tracker: The tracker containing the current state of the conversation.
417
+ slot_name: The name of the slot.
418
+
419
+ Returns:
420
+ The slot value as a string.
421
+ """
422
+ slot_value = tracker.get_slot(slot_name)
423
+ if slot_value is None:
424
+ return "undefined"
425
+ else:
426
+ return str(slot_value)
427
+
428
+ def prepare_current_flow_slots_for_template(
429
+ self, top_flow: Flow, current_step: FlowStep, tracker: DialogueStateTracker
430
+ ) -> List[Dict[str, Any]]:
431
+ """Prepare the current flow slots for the template.
432
+
433
+ Args:
434
+ top_flow: The top flow.
435
+ current_step: The current step in the flow.
436
+ tracker: The tracker containing the current state of the conversation.
437
+
438
+ Returns:
439
+ The slots with values, types, allowed values and a description.
440
+ """
441
+ if top_flow is not None:
442
+ flow_slots = [
443
+ {
444
+ "name": collect_step.collect,
445
+ "value": self.get_slot_value(tracker, collect_step.collect),
446
+ "type": tracker.slots[collect_step.collect].type_name,
447
+ "allowed_values": allowed_values_for_slot(
448
+ tracker.slots[collect_step.collect]
449
+ ),
450
+ "description": collect_step.description,
451
+ }
452
+ for collect_step in top_flow.get_collect_steps()
453
+ if self.is_extractable(collect_step, tracker, current_step)
454
+ ]
455
+ else:
456
+ flow_slots = []
457
+ return flow_slots
458
+
459
+ def prepare_current_slot_for_template(
460
+ self, current_step: FlowStep
461
+ ) -> Tuple[Union[str, None], Union[str, None]]:
462
+ """Prepare the current slot for the template."""
463
+ return (
464
+ (current_step.collect, current_step.description)
465
+ if isinstance(current_step, CollectInformationFlowStep)
466
+ else (None, None)
467
+ )
@@ -0,0 +1,67 @@
1
+ from typing import Dict, Any, Optional, Text
2
+
3
+ import structlog
4
+ from deprecated import deprecated # type: ignore[import]
5
+
6
+ from rasa.dialogue_understanding.generator.single_step.single_step_llm_command_generator import ( # noqa: E501
7
+ SingleStepLLMCommandGenerator,
8
+ )
9
+ from rasa.engine.recipes.default_recipe import DefaultV1Recipe
10
+ from rasa.engine.storage.resource import Resource
11
+ from rasa.engine.storage.storage import ModelStorage
12
+ from rasa.shared.exceptions import ProviderClientAPIException
13
+ from rasa.shared.utils.io import raise_deprecation_warning
14
+
15
+ structlogger = structlog.get_logger()
16
+
17
+
18
+ @DefaultV1Recipe.register(
19
+ [
20
+ DefaultV1Recipe.ComponentType.COMMAND_GENERATOR,
21
+ ],
22
+ is_trainable=True,
23
+ )
24
+ @deprecated(
25
+ reason=(
26
+ "The LLMCommandGenerator is deprecated and will be removed in Rasa 4.0.0. "
27
+ "Please use use SingleStepLLMCommandGenerator instead."
28
+ )
29
+ )
30
+ class LLMCommandGenerator(SingleStepLLMCommandGenerator):
31
+ def __init__(
32
+ self,
33
+ config: Dict[str, Any],
34
+ model_storage: ModelStorage,
35
+ resource: Resource,
36
+ prompt_template: Optional[Text] = None,
37
+ **kwargs: Any,
38
+ ) -> None:
39
+ raise_deprecation_warning(
40
+ message="`LLMCommandGenerator` has been renamed "
41
+ "to `SingleStepLLMCommandGenerator`."
42
+ "Support for the former name `LLMCommandGenerator` "
43
+ "will be removed in Rasa `4.0.0`."
44
+ "Please modify your assistant's configuration to "
45
+ "use `SingleStepLLMCommandGenerator` instead."
46
+ )
47
+
48
+ super().__init__(
49
+ config,
50
+ model_storage,
51
+ resource,
52
+ prompt_template=prompt_template,
53
+ **kwargs,
54
+ )
55
+
56
+ async def invoke_llm(self, prompt: Text) -> Optional[Text]:
57
+ try:
58
+ return await super().invoke_llm(prompt)
59
+ except ProviderClientAPIException:
60
+ # Returning 'None' in case of a ProviderClientAPIException ensures
61
+ # backward compatibility with previous versions. In earlier
62
+ # implementations, 'invoke_llm' was expected to return 'None' when
63
+ # encountering API-related issues. This practice is maintained here
64
+ # to prevent breaking changes in how exceptions are handled and
65
+ # propagated in existing integrations, where callers might not yet
66
+ # be prepared to manage exceptions directly.
67
+ return None
@@ -0,0 +1,62 @@
1
+ {% if flow_active %}
2
+ Your task is to extract slots from a task-oriented conversation. We call these tasks flows.
3
+
4
+ ===
5
+ Here are the available flows, read them carefully. They represent the central business logic:
6
+ {% for flow in available_flows %}
7
+ - {{ flow.name }}: {{ flow.description }}
8
+ {%- endfor %}
9
+ {% else %}
10
+ Extract slots for a business process.
11
+
12
+ The current user message started the process "{{ current_flow }}".
13
+ Here are its slots: {% for slot in flow_slots -%}
14
+ - {{ slot.name }} ({{slot.type}}){% if slot.description %}: {{ slot.description}} {% endif %}{% if slot.allowed_values %}(allowed values: {{ slot.allowed_values }}){% endif %}
15
+ {% endfor %}
16
+ {% endif %}
17
+ ===
18
+ Here is the current conversation:
19
+ {{ current_conversation }}
20
+
21
+ ===
22
+ {% if flow_active %}
23
+ You are currently in the flow {% if top_flow_is_pattern%}"{{ top_user_flow }}"{% else %}"{{ current_flow }}"{% endif %}.
24
+ This flow has the following slots:
25
+ {% for slot in flow_slots -%}
26
+ - {{ slot.name }}
27
+ {% endfor %}{% if top_flow_is_pattern%}{% for slot in top_user_flow_slots -%}
28
+ - {{ slot.name }}
29
+ {% endfor %}{% endif %}
30
+ You have just asked the user for the slot "{{ current_slot }}" ({{current_slot_type}}).
31
+ {% if current_slot_description %}Slot description: "{{ current_slot_description }}"{% endif %}
32
+ {% if current_slot_allowed_values %}Allowed values for this slot: {{ current_slot_allowed_values }}{% endif %}
33
+ The user answered "{{ last_user_message }}".
34
+
35
+ ===
36
+ {% endif %}
37
+ Based on this information, generate a list of actions. These are the available actions:
38
+ 1. Setting or correcting slots, described by "SetSlot(slot_name, slot_value)". An example would be
39
+ "SetSlot(recipient, Freddy)". DO NOT set a slot with an arbitrary value!
40
+ {% if flow_active %}
41
+ 2. Indicating that the users intent goes beyond responding to a question from the AI and setting a slot, described by
42
+ "ChangeFlow()". Add this action, for example, if the user might want to start a different flow, cancel the current one,
43
+ skip a question, ask a question or engages in chitchat. DO NOT add the name of the flow inside "ChangeFlow" command.
44
+ {% else %}
45
+ 2. Finishing your action list, described by "Done()" once everything is taken care of.
46
+
47
+ ===
48
+ Analyze the latest user message and see if there is any additional information the user already provided to fill the
49
+ slots.
50
+ If they did not provide information for the slots already, do not fill them.
51
+ Also, if they had provided some information but had cancelled the process and are now trying to restart the process do
52
+ not fill slots with information given previously when they cancelled the process.
53
+ {% endif %}
54
+ ===
55
+
56
+ Summarize the last user message in the context of the conversation. Then generate a final list of actions.
57
+ ===
58
+ {% if flow_active %}
59
+ The user saying "{{ last_user_message }}" after being asked for the slot "{{ current_slot }}" means that they might
60
+ {% else %}
61
+ The user saying "{{ last_user_message }}" means that they might
62
+ {% endif %}
@@ -0,0 +1,38 @@
1
+ Your task is to analyze the current situation and to start and/or end business processes that we call flows.
2
+
3
+ ===
4
+ Here are the available flows, read them carefully. They represent the central business logic:
5
+ {% for flow in available_flows %}
6
+ - {{ flow.name }}: {{ flow.description }}
7
+ {%- endfor %}
8
+
9
+ ===
10
+ Here is the current conversation:
11
+ {{ current_conversation }}
12
+
13
+ ===
14
+ {% if current_flow != None and not top_flow_is_pattern%}You are currently in the flow "{{ current_flow }}".{% else %}You are currently not in any flow.{% endif %}
15
+
16
+ ===
17
+ Based on this information generate a list of actions, you want to take. Any logic of what happens afterwards is handled
18
+ by the flow engine. These are your available actions:
19
+ 1. Starting a flow, described by "StartFlow(flow_name)".
20
+ 2. If unsure which flow should be started, clarify candidate flows, described by "Clarify(flow_name_1, flow_name_2, ...)". This is particularly useful for short messages.{% if current_flow != None %}
21
+ 3. Stopping the current flow on request of the user, described by "CancelFlow()".{% endif %}
22
+ 4. Indicating that the users intent goes beyond starting, clarifying or canceling a flow, described by "CannotHandle()".
23
+
24
+ ===
25
+ Here are some examples of user messages -> action list:
26
+ 1. "book" -> Clarify(book_hotel, book_restaurant, book_flight)
27
+ 2. "I want to book a hotel in Paris." -> StartFlow(book_hotel)
28
+ 3. "Can you help me with my booking?" -> Clarify(book_hotel, book_restaurant, book_flight)
29
+ 4. "Nevermind, stop it." -> CancelFlow()
30
+ ===
31
+ Write out the actions you want to take for the last user message, one per line.
32
+ Do not take unnecessary actions.
33
+ Strictly adhere to the provided actions and their format listed above.
34
+ Use the previous conversation steps only to aid understanding.
35
+
36
+ Summarize the last user message in the context of the conversation. Then generate a final list of actions.
37
+ ===
38
+ The user saying "{{ last_user_message }}" means that they might