langflow-base-nightly 0.5.1.dev2__py3-none-any.whl → 0.5.1.dev4__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.
Files changed (935) hide show
  1. langflow/__init__.py +215 -0
  2. langflow/__main__.py +16 -2
  3. langflow/alembic/versions/006b3990db50_add_unique_constraints.py +4 -7
  4. langflow/alembic/versions/012fb73ac359_add_folder_table.py +4 -5
  5. langflow/alembic/versions/0ae3a2674f32_update_the_columns_that_need_to_change_.py +11 -20
  6. langflow/alembic/versions/0b8757876a7c_.py +4 -7
  7. langflow/alembic/versions/0d60fcbd4e8e_create_vertex_builds_table.py +4 -6
  8. langflow/alembic/versions/1a110b568907_replace_credential_table_with_variable.py +4 -5
  9. langflow/alembic/versions/1b8b740a6fa3_remove_fk_constraint_in_message_.py +32 -27
  10. langflow/alembic/versions/1c79524817ed_add_unique_constraints_per_user_in_.py +4 -5
  11. langflow/alembic/versions/1d90f8a0efe1_update_description_columns_type.py +4 -5
  12. langflow/alembic/versions/1eab2c3eb45e_event_error.py +14 -15
  13. langflow/alembic/versions/1ef9c4f3765d_.py +5 -10
  14. langflow/alembic/versions/1f4d6df60295_add_default_fields_column.py +4 -5
  15. langflow/alembic/versions/260dbcc8b680_adds_tables.py +4 -5
  16. langflow/alembic/versions/29fe8f1f806b_add_missing_index.py +4 -5
  17. langflow/alembic/versions/2ac71eb9c3ae_adds_credential_table.py +4 -7
  18. langflow/alembic/versions/3bb0ddf32dfb_add_unique_constraints_per_user_in_flow_.py +4 -5
  19. langflow/alembic/versions/4e5980a44eaa_fix_date_times_again.py +1 -2
  20. langflow/alembic/versions/58b28437a398_modify_nullable.py +1 -2
  21. langflow/alembic/versions/5ace73a7f223_new_remove_table_upgrade_op.py +6 -12
  22. langflow/alembic/versions/631faacf5da2_add_webhook_columns.py +4 -5
  23. langflow/alembic/versions/63b9c451fd30_add_icon_and_icon_bg_color_to_flow.py +4 -5
  24. langflow/alembic/versions/66f72f04a1de_add_mcp_support_with_project_settings_.py +21 -23
  25. langflow/alembic/versions/67cc006d50bf_add_profile_image_column.py +4 -5
  26. langflow/alembic/versions/6e7b581b5648_fix_nullable.py +4 -5
  27. langflow/alembic/versions/7843803a87b5_store_updates.py +4 -6
  28. langflow/alembic/versions/79e675cb6752_change_datetime_type.py +1 -2
  29. langflow/alembic/versions/7d2162acc8b2_adds_updated_at_and_folder_cols.py +4 -10
  30. langflow/alembic/versions/90be8e2ed91e_create_transactions_table.py +4 -6
  31. langflow/alembic/versions/93e2705fa8d6_add_column_save_path_to_flow.py +7 -9
  32. langflow/alembic/versions/a72f5cf9c2f9_add_endpoint_name_col.py +4 -5
  33. langflow/alembic/versions/b2fa308044b5_add_unique_constraints.py +1 -2
  34. langflow/alembic/versions/bc2f01c40e4a_new_fixes.py +4 -5
  35. langflow/alembic/versions/c153816fd85f_set_name_and_value_to_not_nullable.py +4 -5
  36. langflow/alembic/versions/d066bfd22890_add_message_table.py +4 -4
  37. langflow/alembic/versions/d2d475a1f7c0_add_tags_column_to_flow.py +12 -13
  38. langflow/alembic/versions/d3dbf656a499_add_gradient_column_in_flow.py +12 -12
  39. langflow/alembic/versions/d9a6ea21edcd_rename_default_folder.py +7 -10
  40. langflow/alembic/versions/dd9e0804ebd1_add_v2_file_table.py +8 -7
  41. langflow/alembic/versions/e3162c1804e6_add_persistent_locked_state.py +10 -10
  42. langflow/alembic/versions/e3bc869fa272_fix_nullable.py +4 -5
  43. langflow/alembic/versions/e56d87f8994a_add_optins_column_to_user.py +13 -14
  44. langflow/alembic/versions/e5a65ecff2cd_nullable_in_vertex_build.py +4 -5
  45. langflow/alembic/versions/eb5866d51fd2_change_columns_to_be_nullable.py +4 -5
  46. langflow/alembic/versions/eb5e72293a8e_add_error_and_edit_flags_to_message.py +4 -5
  47. langflow/alembic/versions/f3b2d1f1002d_add_column_access_type_to_flow.py +19 -15
  48. langflow/alembic/versions/f5ee9749d1a6_user_id_can_be_null_in_flow.py +4 -6
  49. langflow/alembic/versions/fd531f8868b1_fix_credential_table.py +5 -8
  50. langflow/api/build.py +5 -4
  51. langflow/api/health_check_router.py +1 -1
  52. langflow/api/limited_background_tasks.py +1 -1
  53. langflow/api/log_router.py +1 -2
  54. langflow/api/utils.py +2 -2
  55. langflow/api/v1/base.py +1 -2
  56. langflow/api/v1/callback.py +4 -9
  57. langflow/api/v1/chat.py +6 -7
  58. langflow/api/v1/endpoints.py +15 -15
  59. langflow/api/v1/files.py +1 -1
  60. langflow/api/v1/flows.py +1 -1
  61. langflow/api/v1/knowledge_bases.py +1 -1
  62. langflow/api/v1/mcp.py +1 -1
  63. langflow/api/v1/mcp_projects.py +14 -5
  64. langflow/api/v1/mcp_utils.py +3 -3
  65. langflow/api/v1/openai_responses.py +4 -4
  66. langflow/api/v1/schemas.py +3 -38
  67. langflow/api/v1/starter_projects.py +61 -3
  68. langflow/api/v1/store.py +1 -1
  69. langflow/api/v1/validate.py +3 -3
  70. langflow/api/v1/voice_mode.py +2 -2
  71. langflow/api/v2/files.py +1 -1
  72. langflow/api/v2/mcp.py +2 -2
  73. langflow/base/__init__.py +11 -0
  74. langflow/base/agents/__init__.py +3 -0
  75. langflow/base/data/__init__.py +2 -4
  76. langflow/base/data/utils.py +2 -197
  77. langflow/base/embeddings/__init__.py +3 -0
  78. langflow/base/io/__init__.py +7 -0
  79. langflow/base/io/chat.py +5 -18
  80. langflow/base/io/text.py +2 -21
  81. langflow/base/knowledge_bases/__init__.py +3 -0
  82. langflow/base/memory/__init__.py +3 -0
  83. langflow/base/models/__init__.py +2 -2
  84. langflow/base/models/openai_constants.py +6 -120
  85. langflow/base/prompts/__init__.py +3 -0
  86. langflow/base/prompts/api_utils.py +2 -223
  87. langflow/base/textsplitters/__init__.py +3 -0
  88. langflow/base/tools/__init__.py +3 -0
  89. langflow/base/vectorstores/__init__.py +3 -0
  90. langflow/components/__init__.py +7 -259
  91. langflow/components/agents.py +6 -0
  92. langflow/components/anthropic.py +6 -0
  93. langflow/components/data.py +6 -0
  94. langflow/components/helpers.py +6 -0
  95. langflow/components/knowledge_bases/ingestion.py +13 -14
  96. langflow/components/knowledge_bases/retrieval.py +8 -7
  97. langflow/components/openai.py +6 -0
  98. langflow/components/processing/__init__.py +1 -117
  99. langflow/components/processing/converter.py +3 -149
  100. langflow/custom/__init__.py +26 -3
  101. langflow/custom/custom_component/__init__.py +4 -0
  102. langflow/custom/custom_component/component.py +20 -1738
  103. langflow/custom/custom_component/component_with_cache.py +1 -8
  104. langflow/custom/custom_component/custom_component.py +1 -552
  105. langflow/custom/utils.py +1 -872
  106. langflow/custom/validate.py +1 -0
  107. langflow/events/event_manager.py +18 -108
  108. langflow/field_typing/__init__.py +6 -6
  109. langflow/field_typing/constants.py +87 -122
  110. langflow/field_typing/range_spec.py +2 -32
  111. langflow/frontend/assets/{SlackIcon-Cc7Qnzki.js → SlackIcon-v88osOTA.js} +1 -1
  112. langflow/frontend/assets/{Wikipedia-7ulMZY46.js → Wikipedia-DD_S2k00.js} +1 -1
  113. langflow/frontend/assets/{Wolfram-By9PGsHS.js → Wolfram-EO2C5noN.js} +1 -1
  114. langflow/frontend/assets/{index-DVLIDc2_.js → index-1Gv1mfvk.js} +1 -1
  115. langflow/frontend/assets/{index-MVW4HTEk.js → index-7v-bzlzf.js} +1 -1
  116. langflow/frontend/assets/{index-CUzlcce2.js → index-9CbMazbV.js} +1 -1
  117. langflow/frontend/assets/{index-CU16NJD7.js → index-B8ZHP8g2.js} +1 -1
  118. langflow/frontend/assets/{index-v8eXbWlM.js → index-B8y2e6vN.js} +1 -1
  119. langflow/frontend/assets/{index-BX_asvRB.js → index-BBRUGsyr.js} +1 -1
  120. langflow/frontend/assets/{index-9FL5xjkL.js → index-BGwqQwlh.js} +1 -1
  121. langflow/frontend/assets/{index-BAn-AzCS.js → index-BIq-k-FG.js} +1 -1
  122. langflow/frontend/assets/{index-D5c2nNvp.js → index-BSN73YP8.js} +1 -1
  123. langflow/frontend/assets/{index-DMCerPJM.js → index-BU8R8jRn.js} +1 -1
  124. langflow/frontend/assets/{index-CvSoff-8.js → index-BV6yx8ey.js} +1 -1
  125. langflow/frontend/assets/{index-BISPW-f6.js → index-BYIsg-Eh.js} +1 -1
  126. langflow/frontend/assets/{index-GzOGB_fo.js → index-B_ksDBSQ.js} +1 -1
  127. langflow/frontend/assets/{index-BIqEYjNT.js → index-Ba1UOZ9A.js} +1 -1
  128. langflow/frontend/assets/{index-ByxGmq5p.js → index-Ba9tKRQg.js} +1 -1
  129. langflow/frontend/assets/{index-BLEWsL1U.js → index-Bbfaw8ca.js} +1 -1
  130. langflow/frontend/assets/{index-C_MhBX6R.js → index-BbuGqvAx.js} +1 -1
  131. langflow/frontend/assets/{index-RH_I78z_.js → index-BeoXu1YX.js} +1 -1
  132. langflow/frontend/assets/{index-cYFKmtmg.js → index-BfjZmOnH.js} +1 -1
  133. langflow/frontend/assets/{index-Bm9i8F4W.js → index-Bjzy_HZB.js} +1 -1
  134. langflow/frontend/assets/{index-_szO7sta.js → index-BofEkpYB.js} +1 -1
  135. langflow/frontend/assets/{index-DP1oE6QB.js → index-Bp7Mty2H.js} +1 -1
  136. langflow/frontend/assets/{index-CeswGUz3.js → index-BqX1H6yK.js} +1 -1
  137. langflow/frontend/assets/{index-C8pI0lzi.js → index-BqtBAJAN.js} +1 -1
  138. langflow/frontend/assets/{index-BusCv3bR.js → index-Bsfraj7A.js} +1 -1
  139. langflow/frontend/assets/{index-BWnKMRFJ.js → index-BtFl7fER.js} +1 -1
  140. langflow/frontend/assets/{index-DnlVWWU8.js → index-BvX993Sv.js} +1 -1
  141. langflow/frontend/assets/{index-C676MS3I.js → index-BvgQ2vzM.js} +1 -1
  142. langflow/frontend/assets/{index-DJ6HD14g.js → index-BwY98u8n.js} +1 -1
  143. langflow/frontend/assets/{index-C51yNvIL.js → index-C-RIJAOS.js} +1 -1
  144. langflow/frontend/assets/{index-DiblXWmk.js → index-C1K6A38P.js} +1 -1
  145. langflow/frontend/assets/{index-Co__gFM1.js → index-C3Vwhx0t.js} +1 -1
  146. langflow/frontend/assets/{index-Coi86oqP.js → index-C5XUG_gr.js} +1 -1
  147. langflow/frontend/assets/{index-jwzN3Jd_.js → index-C6ouLG9o.js} +1 -1
  148. langflow/frontend/assets/{index-CQQ-4XMS.js → index-C7ZJ_Z6f.js} +1 -1
  149. langflow/frontend/assets/{index-Bl7RpmrB.js → index-CCOGIwGY.js} +1 -1
  150. langflow/frontend/assets/{index-CVkIdc6y.js → index-CCcye2rt.js} +1 -1
  151. langflow/frontend/assets/{index-bMhyLtgS.js → index-CFR4yJQB.js} +1 -1
  152. langflow/frontend/assets/{index-aAgSKWb3.js → index-CIGmPP0H.js} +1 -1
  153. langflow/frontend/assets/{index-BGt6jQ4x.js → index-CJmMEa6d.js} +1 -1
  154. langflow/frontend/assets/{index-DX7JcSMz.js → index-CJxD7lyU.js} +1 -1
  155. langflow/frontend/assets/{index-BZ-A4K98.js → index-CL_vu6ut.js} +1 -1
  156. langflow/frontend/assets/{index-BMpKFGhI.js → index-COf3UnBn.js} +1 -1
  157. langflow/frontend/assets/{index-xN8ogFdo.js → index-CV9650h_.js} +1 -1
  158. langflow/frontend/assets/{index-OsUvqIUr.js → index-CVDzych0.js} +1 -1
  159. langflow/frontend/assets/{index-BH7AyHxp.js → index-CWIHsC4D.js} +1 -1
  160. langflow/frontend/assets/{index-mjwtJmkP.js → index-CXCnFZ0L.js} +1 -1
  161. langflow/frontend/assets/{index-3jlSQi5Y.js → index-Ca_Pw_Dn.js} +1 -1
  162. langflow/frontend/assets/{index-D-SnFlhU.js → index-Cbb3bX9e.js} +1 -1
  163. langflow/frontend/assets/{index--e0oQqZh.js → index-CcJtOz-Z.js} +1 -1
  164. langflow/frontend/assets/{index-S-sc0Cm9.js → index-CfTbTHEv.js} +1 -1
  165. langflow/frontend/assets/{index-Deu8rlaZ.js → index-ChoxDAgX.js} +1 -1
  166. langflow/frontend/assets/{index-lnF9Eqr2.js → index-Cn4gw8aE.js} +1 -1
  167. langflow/frontend/assets/{index-C_NwzK6j.js → index-CnpLg4zX.js} +1 -1
  168. langflow/frontend/assets/{index-DznH7Jbq.js → index-Cpao2omG.js} +1 -1
  169. langflow/frontend/assets/{index-DpWrk8mA.js → index-CqoxM01j.js} +1 -1
  170. langflow/frontend/assets/{index-Bw-TIIC6.js → index-CrHf2Ic1.js} +1 -1
  171. langflow/frontend/assets/{index-DmYLDQag.js → index-CrV0uIjp.js} +1 -1
  172. langflow/frontend/assets/{index-Dp7ZQyL3.js → index-CssADaak.js} +1 -1
  173. langflow/frontend/assets/{index-CNh0rwur.js → index-CtJdNLy9.js} +1 -1
  174. langflow/frontend/assets/{index-Ca1b7Iag.js → index-CyeWD2dh.js} +1 -1
  175. langflow/frontend/assets/{index-DcApTyZ7.js → index-D1xzD7uc.js} +1 -1
  176. langflow/frontend/assets/{index-B3GvPjhD.js → index-D6MuXC4L.js} +1 -1
  177. langflow/frontend/assets/{index-Cw0UComa.js → index-D8w9zvIF.js} +1 -1
  178. langflow/frontend/assets/{index-C-2MRYoJ.js → index-D98Gn0A6.js} +1 -1
  179. langflow/frontend/assets/{index-aWnZIwHd.js → index-DBhjpWkf.js} +1 -1
  180. langflow/frontend/assets/{index-nw3WF9lY.js → index-DCCRJzcY.js} +1 -1
  181. langflow/frontend/assets/{index-RjeC0kaX.js → index-DCTRSkEW.js} +1 -1
  182. langflow/frontend/assets/{index-B_kBTgxV.js → index-DCUfitVj.js} +1 -1
  183. langflow/frontend/assets/{index-ChsGhZn3.js → index-DDdz-Xcl.js} +1 -1
  184. langflow/frontend/assets/{index-7yAHPRxv.js → index-DGdMwZjG.js} +1 -1
  185. langflow/frontend/assets/{index-DjQElpEg.js → index-DGtl2vMw.js} +1 -1
  186. langflow/frontend/assets/{index-BCXhKCOK.js → index-DHVdkrni.js} +1 -1
  187. langflow/frontend/assets/{index-S8uJXTOq.js → index-DJBWwjgl.js} +1 -1
  188. langflow/frontend/assets/{index-qiVTWUuf.js → index-DMAkJ_qX.js} +1 -1
  189. langflow/frontend/assets/{index-D-WStJI6.js → index-DMEvEQI5.js} +1 -1
  190. langflow/frontend/assets/{index-BhqVw9WQ.js → index-DNGRoOsp.js} +1 -1
  191. langflow/frontend/assets/{index-Cu7vC48Y.js → index-DNT_TUTa.js} +1 -1
  192. langflow/frontend/assets/{index-Bhcv5M0n.js → index-DQKOH_9K.js} +1 -1
  193. langflow/frontend/assets/{index-CLcaktde.js → index-DQhqqtqQ.js} +1 -1
  194. langflow/frontend/assets/{index-DZVgPCio.js → index-DRM7KKnG.js} +1 -1
  195. langflow/frontend/assets/{index-uybez8MR.js → index-DSCtl3a5.js} +1 -1
  196. langflow/frontend/assets/{index-CJ5A6STv.js → index-DSLNlm0Z.js} +1 -1
  197. langflow/frontend/assets/{index-Drg8me2a.js → index-DT-PspE-.js} +1 -1
  198. langflow/frontend/assets/{index-DsEZjOcp.js → index-DTpbH-p8.js} +1 -1
  199. langflow/frontend/assets/{index-DrXXKzpD.js → index-DWV6MsIq.js} +1 -1
  200. langflow/frontend/assets/{index-4JIEdyIM.js → index-DWeL4US_.js} +1 -1
  201. langflow/frontend/assets/{index-BlDsBQ_1.js → index-DYKZHhpU.js} +1 -1
  202. langflow/frontend/assets/{index-DFY8YFbC.js → index-DZyQHiMR.js} +1 -1
  203. langflow/frontend/assets/{index-CKPZpkQk.js → index-Dc6qVuSa.js} +1 -1
  204. langflow/frontend/assets/{index-yyAaYjLR.js → index-DkYuicnC.js} +1 -1
  205. langflow/frontend/assets/{index-DmVt5Jlx.js → index-Dlj_2mMs.js} +1 -1
  206. langflow/frontend/assets/{index-BvRIG6P5.js → index-DmGJUrEp.js} +1 -1
  207. langflow/frontend/assets/{index-BWFIrwW1.js → index-Dn6hpCAZ.js} +1 -1
  208. langflow/frontend/assets/{index-Cb5G9Ifd.js → index-DrJU8Fgb.js} +1 -1
  209. langflow/frontend/assets/{index-COoTCxvs.js → index-DsWfdCzp.js} +1 -1
  210. langflow/frontend/assets/{index-ZjeocHyu.js → index-DvCPWs2_.js} +1 -1
  211. langflow/frontend/assets/{index-B5LHnuQR.js → index-DvPVq7OP.js} +1 -1
  212. langflow/frontend/assets/{index-BnCnYnao.js → index-Dw71ufW4.js} +1 -1
  213. langflow/frontend/assets/{index-AALDfCyt.js → index-DxkJactf.js} +1 -1
  214. langflow/frontend/assets/{index-k9jP5chN.js → index-Dz2GTphU.js} +1 -1
  215. langflow/frontend/assets/{index-BdjfHsrf.js → index-Fvd524_c.js} +1 -1
  216. langflow/frontend/assets/{index-AKVkmT4S.js → index-GAQ0Mk2M.js} +1 -1
  217. langflow/frontend/assets/{index-BZSa2qz7.js → index-Hm5-4ItD.js} +1 -1
  218. langflow/frontend/assets/{index-DbfS_UH-.js → index-IT67FzsK.js} +1 -1
  219. langflow/frontend/assets/{index-BLXN681C.js → index-ItYiij1i.js} +1 -1
  220. langflow/frontend/assets/{index-CiklyQU3.js → index-IuR_FEdB.js} +1 -1
  221. langflow/frontend/assets/{index-xV6ystWy.js → index-Jj60FQkv.js} +1 -1
  222. langflow/frontend/assets/{index-C_157Mb-.js → index-LlvshmVz.js} +1 -1
  223. langflow/frontend/assets/{index-CDphUsa3.js → index-LwKh3I_W.js} +1 -1
  224. langflow/frontend/assets/{index-BrDz-PxE.js → index-N-xxmKKH.js} +1 -1
  225. langflow/frontend/assets/{index-BsdLyYMY.js → index-RwpaHIAH.js} +1 -1
  226. langflow/frontend/assets/{index-Cu2Xr6_j.js → index-TVvsp-xh.js} +1 -1
  227. langflow/frontend/assets/{index-CPiM2oyj.js → index-TdE2u9zP.js} +1 -1
  228. langflow/frontend/assets/{index-DOj_QWqG.js → index-_x-NkYeW.js} +1 -1
  229. langflow/frontend/assets/{index-YJsAl7vm.js → index-a-YclEbW.js} +1 -1
  230. langflow/frontend/assets/{index-5-CSw2-z.js → index-e9MFKUCo.js} +1 -1
  231. langflow/frontend/assets/{index-BSwBVwyF.js → index-krPr8f2F.js} +1 -1
  232. langflow/frontend/assets/{index-Df6psZEj.js → index-kveiUWuL.js} +1 -1
  233. langflow/frontend/assets/{index-CF4_Og1m.js → index-lE3oSjJi.js} +1 -1
  234. langflow/frontend/assets/{index-C6nzdeYx.js → index-lM3UYg7F.js} +1 -1
  235. langflow/frontend/assets/{index-C-wnbBBY.js → index-nsRk3qgA.js} +1 -1
  236. langflow/frontend/assets/{index-D234yKNJ.js → index-pBO0SZLD.js} +4 -4
  237. langflow/frontend/assets/{index-BMvp94tO.js → index-pbZHsbuE.js} +1 -1
  238. langflow/frontend/assets/{index-hg2y9OAt.js → index-sfX3aWyp.js} +1 -1
  239. langflow/frontend/assets/{index-DTCrijba.js → index-xQz-VJ0-.js} +1 -1
  240. langflow/frontend/assets/{index-SB4rw8D5.js → index-yfcsaHS6.js} +1 -1
  241. langflow/frontend/assets/{index-C-bjC2sz.js → index-zcGjo9fx.js} +1 -1
  242. langflow/frontend/assets/lazyIconImports-BjqDmNYG.js +2 -0
  243. langflow/frontend/assets/{use-post-add-user-JUeLDErC.js → use-post-add-user-w3vpKSOB.js} +1 -1
  244. langflow/frontend/index.html +1 -1
  245. langflow/graph/__init__.py +4 -4
  246. langflow/helpers/data.py +2 -2
  247. langflow/helpers/flow.py +9 -7
  248. langflow/helpers/user.py +2 -2
  249. langflow/initial_setup/setup.py +9 -9
  250. langflow/initial_setup/starter_projects/Basic Prompt Chaining.json +119 -41
  251. langflow/initial_setup/starter_projects/Basic Prompting.json +45 -19
  252. langflow/initial_setup/starter_projects/Blog Writer.json +53 -21
  253. langflow/initial_setup/starter_projects/Custom Component Generator.json +121 -97
  254. langflow/initial_setup/starter_projects/Document Q&A.json +46 -18
  255. langflow/initial_setup/starter_projects/Financial Report Parser.json +49 -17
  256. langflow/initial_setup/starter_projects/Hybrid Search RAG.json +89 -50
  257. langflow/initial_setup/starter_projects/Image Sentiment Analysis.json +86 -22
  258. langflow/initial_setup/starter_projects/Instagram Copywriter.json +210 -57
  259. langflow/initial_setup/starter_projects/Invoice Summarizer.json +132 -35
  260. langflow/initial_setup/starter_projects/Knowledge Ingestion.json +8 -8
  261. langflow/initial_setup/starter_projects/Knowledge Retrieval.json +8 -8
  262. langflow/initial_setup/starter_projects/Market Research.json +174 -48
  263. langflow/initial_setup/starter_projects/Meeting Summary.json +102 -38
  264. langflow/initial_setup/starter_projects/Memory Chatbot.json +49 -21
  265. langflow/initial_setup/starter_projects/News Aggregator.json +140 -39
  266. langflow/initial_setup/starter_projects/Nvidia Remix.json +153 -181
  267. langflow/initial_setup/starter_projects/Pok/303/251dex Agent.json" +132 -35
  268. langflow/initial_setup/starter_projects/Portfolio Website Code Generator.json +106 -43
  269. langflow/initial_setup/starter_projects/Price Deal Finder.json +136 -39
  270. langflow/initial_setup/starter_projects/Research Agent.json +206 -53
  271. langflow/initial_setup/starter_projects/Research Translation Loop.json +66 -34
  272. langflow/initial_setup/starter_projects/SEO Keyword Generator.json +41 -15
  273. langflow/initial_setup/starter_projects/SaaS Pricing.json +128 -31
  274. langflow/initial_setup/starter_projects/Search agent.json +132 -35
  275. langflow/initial_setup/starter_projects/Sequential Tasks Agents.json +422 -98
  276. langflow/initial_setup/starter_projects/Simple Agent.json +150 -42
  277. langflow/initial_setup/starter_projects/Social Media Agent.json +150 -42
  278. langflow/initial_setup/starter_projects/Text Sentiment Analysis.json +120 -24
  279. langflow/initial_setup/starter_projects/Travel Planning Agents.json +418 -94
  280. langflow/initial_setup/starter_projects/Twitter Thread Generator.json +69 -37
  281. langflow/initial_setup/starter_projects/Vector Store RAG.json +66 -38
  282. langflow/initial_setup/starter_projects/Youtube Analysis.json +191 -51
  283. langflow/initial_setup/starter_projects/basic_prompting.py +4 -4
  284. langflow/initial_setup/starter_projects/blog_writer.py +5 -5
  285. langflow/initial_setup/starter_projects/complex_agent.py +8 -8
  286. langflow/initial_setup/starter_projects/document_qa.py +5 -5
  287. langflow/initial_setup/starter_projects/hierarchical_tasks_agent.py +8 -8
  288. langflow/initial_setup/starter_projects/memory_chatbot.py +6 -6
  289. langflow/initial_setup/starter_projects/sequential_tasks_agent.py +7 -7
  290. langflow/initial_setup/starter_projects/vector_store_rag.py +8 -8
  291. langflow/inputs/__init__.py +3 -2
  292. langflow/inputs/constants.py +3 -2
  293. langflow/inputs/input_mixin.py +49 -310
  294. langflow/inputs/inputs.py +72 -703
  295. langflow/inputs/validators.py +2 -18
  296. langflow/interface/__init__.py +4 -0
  297. langflow/interface/components.py +3 -491
  298. langflow/interface/initialize/loading.py +7 -6
  299. langflow/interface/listing.py +3 -25
  300. langflow/interface/run.py +1 -1
  301. langflow/interface/utils.py +3 -111
  302. langflow/io/__init__.py +2 -2
  303. langflow/io/schema.py +11 -302
  304. langflow/load/__init__.py +4 -2
  305. langflow/load/utils.py +2 -96
  306. langflow/logging/__init__.py +2 -1
  307. langflow/logging/setup.py +1 -1
  308. langflow/main.py +8 -5
  309. langflow/memory.py +12 -6
  310. langflow/middleware.py +1 -1
  311. langflow/processing/process.py +7 -7
  312. langflow/schema/__init__.py +22 -5
  313. langflow/schema/artifact.py +1 -1
  314. langflow/schema/data.py +5 -303
  315. langflow/schema/dataframe.py +2 -205
  316. langflow/schema/graph.py +4 -45
  317. langflow/schema/image.py +2 -67
  318. langflow/schema/message.py +6 -470
  319. langflow/schema/playground_events.py +5 -6
  320. langflow/schema/schema.py +24 -117
  321. langflow/serialization/constants.py +3 -2
  322. langflow/serialization/serialization.py +1 -1
  323. langflow/server.py +1 -2
  324. langflow/services/__init__.py +1 -2
  325. langflow/services/auth/mcp_encryption.py +1 -1
  326. langflow/services/auth/service.py +1 -1
  327. langflow/services/auth/utils.py +5 -5
  328. langflow/services/cache/disk.py +2 -2
  329. langflow/services/cache/factory.py +2 -2
  330. langflow/services/cache/service.py +2 -2
  331. langflow/services/cache/utils.py +0 -11
  332. langflow/services/database/factory.py +1 -1
  333. langflow/services/database/models/flow/model.py +1 -1
  334. langflow/services/database/models/message/crud.py +2 -1
  335. langflow/services/database/models/transactions/crud.py +1 -1
  336. langflow/services/database/models/user/crud.py +1 -1
  337. langflow/services/database/service.py +2 -2
  338. langflow/services/database/utils.py +1 -2
  339. langflow/services/deps.py +12 -17
  340. langflow/services/enhanced_manager.py +71 -0
  341. langflow/services/factory.py +14 -7
  342. langflow/services/flow/flow_runner.py +4 -4
  343. langflow/services/job_queue/service.py +2 -1
  344. langflow/services/manager.py +14 -130
  345. langflow/services/schema.py +0 -1
  346. langflow/services/session/service.py +3 -2
  347. langflow/services/settings/__init__.py +0 -3
  348. langflow/services/settings/base.py +16 -549
  349. langflow/services/settings/factory.py +2 -21
  350. langflow/services/settings/feature_flags.py +2 -11
  351. langflow/services/settings/service.py +2 -31
  352. langflow/services/shared_component_cache/factory.py +1 -1
  353. langflow/services/socket/service.py +1 -1
  354. langflow/services/socket/utils.py +1 -8
  355. langflow/services/state/factory.py +1 -1
  356. langflow/services/state/service.py +3 -2
  357. langflow/services/storage/factory.py +2 -2
  358. langflow/services/storage/local.py +1 -2
  359. langflow/services/storage/s3.py +1 -2
  360. langflow/services/storage/service.py +2 -1
  361. langflow/services/store/factory.py +1 -1
  362. langflow/services/store/service.py +2 -2
  363. langflow/services/store/utils.py +1 -2
  364. langflow/services/task/service.py +2 -1
  365. langflow/services/task/temp_flow_cleanup.py +1 -1
  366. langflow/services/telemetry/factory.py +1 -1
  367. langflow/services/telemetry/service.py +2 -3
  368. langflow/services/tracing/arize_phoenix.py +3 -3
  369. langflow/services/tracing/base.py +1 -1
  370. langflow/services/tracing/factory.py +1 -1
  371. langflow/services/tracing/langfuse.py +2 -2
  372. langflow/services/tracing/langsmith.py +2 -2
  373. langflow/services/tracing/langwatch.py +4 -4
  374. langflow/services/tracing/opik.py +2 -2
  375. langflow/services/tracing/service.py +17 -11
  376. langflow/services/tracing/traceloop.py +2 -2
  377. langflow/services/tracing/utils.py +1 -1
  378. langflow/services/utils.py +54 -9
  379. langflow/services/variable/factory.py +1 -1
  380. langflow/services/variable/kubernetes.py +2 -3
  381. langflow/services/variable/kubernetes_secrets.py +1 -2
  382. langflow/services/variable/service.py +2 -3
  383. langflow/template/__init__.py +2 -9
  384. langflow/template/field/__init__.py +3 -0
  385. langflow/template/field/base.py +2 -256
  386. langflow/template/frontend_node.py +3 -0
  387. langflow/template/utils.py +2 -216
  388. langflow/utils/constants.py +28 -204
  389. langflow/utils/lazy_load.py +3 -14
  390. langflow/utils/schemas.py +2 -3
  391. langflow/utils/template_validation.py +2 -2
  392. langflow/utils/util.py +59 -479
  393. langflow/utils/validate.py +2 -488
  394. langflow/utils/voice_utils.py +1 -2
  395. langflow/worker.py +1 -1
  396. {langflow_base_nightly-0.5.1.dev2.dist-info → langflow_base_nightly-0.5.1.dev4.dist-info}/METADATA +2 -1
  397. langflow_base_nightly-0.5.1.dev4.dist-info/RECORD +633 -0
  398. langflow/base/agents/agent.py +0 -267
  399. langflow/base/agents/callback.py +0 -130
  400. langflow/base/agents/context.py +0 -109
  401. langflow/base/agents/crewai/__init__.py +0 -0
  402. langflow/base/agents/crewai/crew.py +0 -231
  403. langflow/base/agents/crewai/tasks.py +0 -12
  404. langflow/base/agents/default_prompts.py +0 -23
  405. langflow/base/agents/errors.py +0 -15
  406. langflow/base/agents/events.py +0 -346
  407. langflow/base/agents/utils.py +0 -205
  408. langflow/base/astra_assistants/__init__.py +0 -0
  409. langflow/base/astra_assistants/util.py +0 -171
  410. langflow/base/chains/__init__.py +0 -0
  411. langflow/base/chains/model.py +0 -19
  412. langflow/base/composio/__init__.py +0 -0
  413. langflow/base/composio/composio_base.py +0 -1297
  414. langflow/base/compressors/__init__.py +0 -0
  415. langflow/base/compressors/model.py +0 -60
  416. langflow/base/constants.py +0 -46
  417. langflow/base/curl/__init__.py +0 -0
  418. langflow/base/curl/parse.py +0 -188
  419. langflow/base/data/base_file.py +0 -685
  420. langflow/base/data/docling_utils.py +0 -245
  421. langflow/base/document_transformers/__init__.py +0 -0
  422. langflow/base/document_transformers/model.py +0 -43
  423. langflow/base/embeddings/aiml_embeddings.py +0 -62
  424. langflow/base/embeddings/model.py +0 -26
  425. langflow/base/flow_processing/__init__.py +0 -0
  426. langflow/base/flow_processing/utils.py +0 -86
  427. langflow/base/huggingface/__init__.py +0 -0
  428. langflow/base/huggingface/model_bridge.py +0 -133
  429. langflow/base/langchain_utilities/__init__.py +0 -0
  430. langflow/base/langchain_utilities/model.py +0 -35
  431. langflow/base/langchain_utilities/spider_constants.py +0 -1
  432. langflow/base/langwatch/__init__.py +0 -0
  433. langflow/base/langwatch/utils.py +0 -18
  434. langflow/base/mcp/__init__.py +0 -0
  435. langflow/base/mcp/constants.py +0 -2
  436. langflow/base/mcp/util.py +0 -1524
  437. langflow/base/memory/memory.py +0 -49
  438. langflow/base/memory/model.py +0 -38
  439. langflow/base/models/aiml_constants.py +0 -51
  440. langflow/base/models/anthropic_constants.py +0 -47
  441. langflow/base/models/aws_constants.py +0 -151
  442. langflow/base/models/chat_result.py +0 -76
  443. langflow/base/models/google_generative_ai_constants.py +0 -70
  444. langflow/base/models/groq_constants.py +0 -134
  445. langflow/base/models/model.py +0 -375
  446. langflow/base/models/model_input_constants.py +0 -299
  447. langflow/base/models/model_metadata.py +0 -41
  448. langflow/base/models/model_utils.py +0 -8
  449. langflow/base/models/novita_constants.py +0 -35
  450. langflow/base/models/ollama_constants.py +0 -49
  451. langflow/base/models/sambanova_constants.py +0 -18
  452. langflow/base/processing/__init__.py +0 -0
  453. langflow/base/prompts/utils.py +0 -61
  454. langflow/base/textsplitters/model.py +0 -28
  455. langflow/base/tools/base.py +0 -26
  456. langflow/base/tools/component_tool.py +0 -324
  457. langflow/base/tools/constants.py +0 -49
  458. langflow/base/tools/flow_tool.py +0 -131
  459. langflow/base/tools/run_flow.py +0 -227
  460. langflow/base/vectorstores/model.py +0 -193
  461. langflow/base/vectorstores/utils.py +0 -22
  462. langflow/base/vectorstores/vector_store_connection_decorator.py +0 -52
  463. langflow/components/FAISS/__init__.py +0 -34
  464. langflow/components/FAISS/faiss.py +0 -111
  465. langflow/components/Notion/__init__.py +0 -19
  466. langflow/components/Notion/add_content_to_page.py +0 -269
  467. langflow/components/Notion/create_page.py +0 -94
  468. langflow/components/Notion/list_database_properties.py +0 -68
  469. langflow/components/Notion/list_pages.py +0 -122
  470. langflow/components/Notion/list_users.py +0 -77
  471. langflow/components/Notion/page_content_viewer.py +0 -93
  472. langflow/components/Notion/search.py +0 -111
  473. langflow/components/Notion/update_page_property.py +0 -114
  474. langflow/components/_importing.py +0 -37
  475. langflow/components/agentql/__init__.py +0 -3
  476. langflow/components/agentql/agentql_api.py +0 -151
  477. langflow/components/agents/__init__.py +0 -4
  478. langflow/components/agents/agent.py +0 -554
  479. langflow/components/agents/mcp_component.py +0 -501
  480. langflow/components/aiml/__init__.py +0 -37
  481. langflow/components/aiml/aiml.py +0 -112
  482. langflow/components/aiml/aiml_embeddings.py +0 -37
  483. langflow/components/amazon/__init__.py +0 -36
  484. langflow/components/amazon/amazon_bedrock_embedding.py +0 -109
  485. langflow/components/amazon/amazon_bedrock_model.py +0 -124
  486. langflow/components/amazon/s3_bucket_uploader.py +0 -211
  487. langflow/components/anthropic/__init__.py +0 -34
  488. langflow/components/anthropic/anthropic.py +0 -187
  489. langflow/components/apify/__init__.py +0 -5
  490. langflow/components/apify/apify_actor.py +0 -325
  491. langflow/components/arxiv/__init__.py +0 -3
  492. langflow/components/arxiv/arxiv.py +0 -163
  493. langflow/components/assemblyai/__init__.py +0 -46
  494. langflow/components/assemblyai/assemblyai_get_subtitles.py +0 -83
  495. langflow/components/assemblyai/assemblyai_lemur.py +0 -183
  496. langflow/components/assemblyai/assemblyai_list_transcripts.py +0 -95
  497. langflow/components/assemblyai/assemblyai_poll_transcript.py +0 -72
  498. langflow/components/assemblyai/assemblyai_start_transcript.py +0 -188
  499. langflow/components/azure/__init__.py +0 -37
  500. langflow/components/azure/azure_openai.py +0 -95
  501. langflow/components/azure/azure_openai_embeddings.py +0 -83
  502. langflow/components/baidu/__init__.py +0 -32
  503. langflow/components/baidu/baidu_qianfan_chat.py +0 -113
  504. langflow/components/bing/__init__.py +0 -3
  505. langflow/components/bing/bing_search_api.py +0 -61
  506. langflow/components/cassandra/__init__.py +0 -40
  507. langflow/components/cassandra/cassandra.py +0 -264
  508. langflow/components/cassandra/cassandra_chat.py +0 -92
  509. langflow/components/cassandra/cassandra_graph.py +0 -238
  510. langflow/components/chains/__init__.py +0 -0
  511. langflow/components/chroma/__init__.py +0 -34
  512. langflow/components/chroma/chroma.py +0 -167
  513. langflow/components/cleanlab/__init__.py +0 -40
  514. langflow/components/cleanlab/cleanlab_evaluator.py +0 -157
  515. langflow/components/cleanlab/cleanlab_rag_evaluator.py +0 -254
  516. langflow/components/cleanlab/cleanlab_remediator.py +0 -131
  517. langflow/components/clickhouse/__init__.py +0 -34
  518. langflow/components/clickhouse/clickhouse.py +0 -135
  519. langflow/components/cloudflare/__init__.py +0 -32
  520. langflow/components/cloudflare/cloudflare.py +0 -81
  521. langflow/components/cohere/__init__.py +0 -40
  522. langflow/components/cohere/cohere_embeddings.py +0 -81
  523. langflow/components/cohere/cohere_models.py +0 -46
  524. langflow/components/cohere/cohere_rerank.py +0 -51
  525. langflow/components/composio/__init__.py +0 -73
  526. langflow/components/composio/composio_api.py +0 -268
  527. langflow/components/composio/dropbox_compnent.py +0 -11
  528. langflow/components/composio/github_composio.py +0 -11
  529. langflow/components/composio/gmail_composio.py +0 -38
  530. langflow/components/composio/googlecalendar_composio.py +0 -11
  531. langflow/components/composio/googlemeet_composio.py +0 -11
  532. langflow/components/composio/googletasks_composio.py +0 -8
  533. langflow/components/composio/linear_composio.py +0 -11
  534. langflow/components/composio/outlook_composio.py +0 -11
  535. langflow/components/composio/reddit_composio.py +0 -11
  536. langflow/components/composio/slack_composio.py +0 -11
  537. langflow/components/composio/slackbot_composio.py +0 -11
  538. langflow/components/composio/supabase_composio.py +0 -11
  539. langflow/components/composio/todoist_composio.py +0 -11
  540. langflow/components/composio/youtube_composio.py +0 -11
  541. langflow/components/confluence/__init__.py +0 -3
  542. langflow/components/confluence/confluence.py +0 -84
  543. langflow/components/couchbase/__init__.py +0 -34
  544. langflow/components/couchbase/couchbase.py +0 -102
  545. langflow/components/crewai/__init__.py +0 -49
  546. langflow/components/crewai/crewai.py +0 -107
  547. langflow/components/crewai/hierarchical_crew.py +0 -46
  548. langflow/components/crewai/hierarchical_task.py +0 -44
  549. langflow/components/crewai/sequential_crew.py +0 -52
  550. langflow/components/crewai/sequential_task.py +0 -73
  551. langflow/components/crewai/sequential_task_agent.py +0 -143
  552. langflow/components/custom_component/__init__.py +0 -34
  553. langflow/components/custom_component/custom_component.py +0 -31
  554. langflow/components/data/__init__.py +0 -25
  555. langflow/components/data/api_request.py +0 -545
  556. langflow/components/data/csv_to_data.py +0 -95
  557. langflow/components/data/directory.py +0 -113
  558. langflow/components/data/file.py +0 -586
  559. langflow/components/data/json_to_data.py +0 -98
  560. langflow/components/data/news_search.py +0 -164
  561. langflow/components/data/rss.py +0 -69
  562. langflow/components/data/sql_executor.py +0 -99
  563. langflow/components/data/url.py +0 -299
  564. langflow/components/data/web_search.py +0 -112
  565. langflow/components/data/webhook.py +0 -56
  566. langflow/components/datastax/__init__.py +0 -70
  567. langflow/components/datastax/astra_assistant_manager.py +0 -306
  568. langflow/components/datastax/astra_db.py +0 -69
  569. langflow/components/datastax/astra_vectorize.py +0 -124
  570. langflow/components/datastax/astradb_cql.py +0 -314
  571. langflow/components/datastax/astradb_graph.py +0 -319
  572. langflow/components/datastax/astradb_tool.py +0 -414
  573. langflow/components/datastax/astradb_vectorstore.py +0 -1285
  574. langflow/components/datastax/create_assistant.py +0 -58
  575. langflow/components/datastax/create_thread.py +0 -32
  576. langflow/components/datastax/dotenv.py +0 -35
  577. langflow/components/datastax/get_assistant.py +0 -37
  578. langflow/components/datastax/getenvvar.py +0 -30
  579. langflow/components/datastax/graph_rag.py +0 -141
  580. langflow/components/datastax/hcd.py +0 -314
  581. langflow/components/datastax/list_assistants.py +0 -25
  582. langflow/components/datastax/run.py +0 -89
  583. langflow/components/deactivated/__init__.py +0 -19
  584. langflow/components/deactivated/amazon_kendra.py +0 -66
  585. langflow/components/deactivated/chat_litellm_model.py +0 -158
  586. langflow/components/deactivated/code_block_extractor.py +0 -26
  587. langflow/components/deactivated/documents_to_data.py +0 -22
  588. langflow/components/deactivated/embed.py +0 -16
  589. langflow/components/deactivated/extract_key_from_data.py +0 -46
  590. langflow/components/deactivated/json_document_builder.py +0 -59
  591. langflow/components/deactivated/list_flows.py +0 -20
  592. langflow/components/deactivated/mcp_sse.py +0 -61
  593. langflow/components/deactivated/mcp_stdio.py +0 -62
  594. langflow/components/deactivated/merge_data.py +0 -93
  595. langflow/components/deactivated/message.py +0 -37
  596. langflow/components/deactivated/metal.py +0 -54
  597. langflow/components/deactivated/multi_query.py +0 -59
  598. langflow/components/deactivated/retriever.py +0 -43
  599. langflow/components/deactivated/selective_passthrough.py +0 -77
  600. langflow/components/deactivated/should_run_next.py +0 -40
  601. langflow/components/deactivated/split_text.py +0 -63
  602. langflow/components/deactivated/store_message.py +0 -24
  603. langflow/components/deactivated/sub_flow.py +0 -124
  604. langflow/components/deactivated/vectara_self_query.py +0 -76
  605. langflow/components/deactivated/vector_store.py +0 -24
  606. langflow/components/deepseek/__init__.py +0 -34
  607. langflow/components/deepseek/deepseek.py +0 -136
  608. langflow/components/docling/__init__.py +0 -43
  609. langflow/components/docling/chunk_docling_document.py +0 -186
  610. langflow/components/docling/docling_inline.py +0 -235
  611. langflow/components/docling/docling_remote.py +0 -193
  612. langflow/components/docling/export_docling_document.py +0 -117
  613. langflow/components/documentloaders/__init__.py +0 -0
  614. langflow/components/duckduckgo/__init__.py +0 -3
  615. langflow/components/duckduckgo/duck_duck_go_search_run.py +0 -92
  616. langflow/components/elastic/__init__.py +0 -37
  617. langflow/components/elastic/elasticsearch.py +0 -267
  618. langflow/components/elastic/opensearch.py +0 -243
  619. langflow/components/embeddings/__init__.py +0 -37
  620. langflow/components/embeddings/similarity.py +0 -76
  621. langflow/components/embeddings/text_embedder.py +0 -64
  622. langflow/components/exa/__init__.py +0 -3
  623. langflow/components/exa/exa_search.py +0 -68
  624. langflow/components/firecrawl/__init__.py +0 -43
  625. langflow/components/firecrawl/firecrawl_crawl_api.py +0 -88
  626. langflow/components/firecrawl/firecrawl_extract_api.py +0 -136
  627. langflow/components/firecrawl/firecrawl_map_api.py +0 -89
  628. langflow/components/firecrawl/firecrawl_scrape_api.py +0 -73
  629. langflow/components/git/__init__.py +0 -4
  630. langflow/components/git/git.py +0 -262
  631. langflow/components/git/gitextractor.py +0 -196
  632. langflow/components/glean/__init__.py +0 -3
  633. langflow/components/glean/glean_search_api.py +0 -173
  634. langflow/components/google/__init__.py +0 -17
  635. langflow/components/google/gmail.py +0 -192
  636. langflow/components/google/google_bq_sql_executor.py +0 -157
  637. langflow/components/google/google_drive.py +0 -92
  638. langflow/components/google/google_drive_search.py +0 -152
  639. langflow/components/google/google_generative_ai.py +0 -147
  640. langflow/components/google/google_generative_ai_embeddings.py +0 -141
  641. langflow/components/google/google_oauth_token.py +0 -89
  642. langflow/components/google/google_search_api_core.py +0 -68
  643. langflow/components/google/google_serper_api_core.py +0 -74
  644. langflow/components/groq/__init__.py +0 -34
  645. langflow/components/groq/groq.py +0 -140
  646. langflow/components/helpers/__init__.py +0 -52
  647. langflow/components/helpers/calculator_core.py +0 -89
  648. langflow/components/helpers/create_list.py +0 -40
  649. langflow/components/helpers/current_date.py +0 -42
  650. langflow/components/helpers/id_generator.py +0 -42
  651. langflow/components/helpers/memory.py +0 -251
  652. langflow/components/helpers/output_parser.py +0 -45
  653. langflow/components/helpers/store_message.py +0 -90
  654. langflow/components/homeassistant/__init__.py +0 -7
  655. langflow/components/homeassistant/home_assistant_control.py +0 -152
  656. langflow/components/homeassistant/list_home_assistant_states.py +0 -137
  657. langflow/components/huggingface/__init__.py +0 -37
  658. langflow/components/huggingface/huggingface.py +0 -197
  659. langflow/components/huggingface/huggingface_inference_api.py +0 -106
  660. langflow/components/ibm/__init__.py +0 -34
  661. langflow/components/ibm/watsonx.py +0 -203
  662. langflow/components/ibm/watsonx_embeddings.py +0 -135
  663. langflow/components/icosacomputing/__init__.py +0 -5
  664. langflow/components/icosacomputing/combinatorial_reasoner.py +0 -84
  665. langflow/components/input_output/__init__.py +0 -38
  666. langflow/components/input_output/chat.py +0 -120
  667. langflow/components/input_output/chat_output.py +0 -200
  668. langflow/components/input_output/text.py +0 -27
  669. langflow/components/input_output/text_output.py +0 -29
  670. langflow/components/jigsawstack/__init__.py +0 -23
  671. langflow/components/jigsawstack/ai_scrape.py +0 -126
  672. langflow/components/jigsawstack/ai_web_search.py +0 -136
  673. langflow/components/jigsawstack/file_read.py +0 -115
  674. langflow/components/jigsawstack/file_upload.py +0 -94
  675. langflow/components/jigsawstack/image_generation.py +0 -205
  676. langflow/components/jigsawstack/nsfw.py +0 -60
  677. langflow/components/jigsawstack/object_detection.py +0 -124
  678. langflow/components/jigsawstack/sentiment.py +0 -112
  679. langflow/components/jigsawstack/text_to_sql.py +0 -90
  680. langflow/components/jigsawstack/text_translate.py +0 -77
  681. langflow/components/jigsawstack/vocr.py +0 -107
  682. langflow/components/langchain_utilities/__init__.py +0 -109
  683. langflow/components/langchain_utilities/character.py +0 -53
  684. langflow/components/langchain_utilities/conversation.py +0 -52
  685. langflow/components/langchain_utilities/csv_agent.py +0 -107
  686. langflow/components/langchain_utilities/fake_embeddings.py +0 -26
  687. langflow/components/langchain_utilities/html_link_extractor.py +0 -35
  688. langflow/components/langchain_utilities/json_agent.py +0 -45
  689. langflow/components/langchain_utilities/langchain_hub.py +0 -126
  690. langflow/components/langchain_utilities/language_recursive.py +0 -49
  691. langflow/components/langchain_utilities/language_semantic.py +0 -138
  692. langflow/components/langchain_utilities/llm_checker.py +0 -39
  693. langflow/components/langchain_utilities/llm_math.py +0 -42
  694. langflow/components/langchain_utilities/natural_language.py +0 -61
  695. langflow/components/langchain_utilities/openai_tools.py +0 -53
  696. langflow/components/langchain_utilities/openapi.py +0 -48
  697. langflow/components/langchain_utilities/recursive_character.py +0 -60
  698. langflow/components/langchain_utilities/retrieval_qa.py +0 -83
  699. langflow/components/langchain_utilities/runnable_executor.py +0 -137
  700. langflow/components/langchain_utilities/self_query.py +0 -80
  701. langflow/components/langchain_utilities/spider.py +0 -142
  702. langflow/components/langchain_utilities/sql.py +0 -40
  703. langflow/components/langchain_utilities/sql_database.py +0 -35
  704. langflow/components/langchain_utilities/sql_generator.py +0 -78
  705. langflow/components/langchain_utilities/tool_calling.py +0 -59
  706. langflow/components/langchain_utilities/vector_store_info.py +0 -49
  707. langflow/components/langchain_utilities/vector_store_router.py +0 -33
  708. langflow/components/langchain_utilities/xml_agent.py +0 -71
  709. langflow/components/langwatch/__init__.py +0 -3
  710. langflow/components/langwatch/langwatch.py +0 -278
  711. langflow/components/link_extractors/__init__.py +0 -0
  712. langflow/components/lmstudio/__init__.py +0 -34
  713. langflow/components/lmstudio/lmstudioembeddings.py +0 -89
  714. langflow/components/lmstudio/lmstudiomodel.py +0 -129
  715. langflow/components/logic/__init__.py +0 -52
  716. langflow/components/logic/conditional_router.py +0 -171
  717. langflow/components/logic/data_conditional_router.py +0 -125
  718. langflow/components/logic/flow_tool.py +0 -110
  719. langflow/components/logic/listen.py +0 -29
  720. langflow/components/logic/loop.py +0 -125
  721. langflow/components/logic/notify.py +0 -88
  722. langflow/components/logic/pass_message.py +0 -35
  723. langflow/components/logic/run_flow.py +0 -71
  724. langflow/components/logic/sub_flow.py +0 -114
  725. langflow/components/maritalk/__init__.py +0 -32
  726. langflow/components/maritalk/maritalk.py +0 -52
  727. langflow/components/mem0/__init__.py +0 -3
  728. langflow/components/mem0/mem0_chat_memory.py +0 -136
  729. langflow/components/milvus/__init__.py +0 -34
  730. langflow/components/milvus/milvus.py +0 -115
  731. langflow/components/mistral/__init__.py +0 -37
  732. langflow/components/mistral/mistral.py +0 -114
  733. langflow/components/mistral/mistral_embeddings.py +0 -58
  734. langflow/components/models/__init__.py +0 -34
  735. langflow/components/models/embedding_model.py +0 -114
  736. langflow/components/models/language_model.py +0 -144
  737. langflow/components/mongodb/__init__.py +0 -34
  738. langflow/components/mongodb/mongodb_atlas.py +0 -213
  739. langflow/components/needle/__init__.py +0 -3
  740. langflow/components/needle/needle.py +0 -104
  741. langflow/components/notdiamond/__init__.py +0 -36
  742. langflow/components/notdiamond/notdiamond.py +0 -228
  743. langflow/components/novita/__init__.py +0 -32
  744. langflow/components/novita/novita.py +0 -130
  745. langflow/components/nvidia/__init__.py +0 -57
  746. langflow/components/nvidia/nvidia.py +0 -157
  747. langflow/components/nvidia/nvidia_embedding.py +0 -77
  748. langflow/components/nvidia/nvidia_ingest.py +0 -317
  749. langflow/components/nvidia/nvidia_rerank.py +0 -63
  750. langflow/components/nvidia/system_assist.py +0 -65
  751. langflow/components/olivya/__init__.py +0 -3
  752. langflow/components/olivya/olivya.py +0 -116
  753. langflow/components/ollama/__init__.py +0 -37
  754. langflow/components/ollama/ollama.py +0 -330
  755. langflow/components/ollama/ollama_embeddings.py +0 -106
  756. langflow/components/openai/__init__.py +0 -37
  757. langflow/components/openai/openai.py +0 -100
  758. langflow/components/openai/openai_chat_model.py +0 -158
  759. langflow/components/openrouter/__init__.py +0 -32
  760. langflow/components/openrouter/openrouter.py +0 -202
  761. langflow/components/output_parsers/__init__.py +0 -0
  762. langflow/components/perplexity/__init__.py +0 -34
  763. langflow/components/perplexity/perplexity.py +0 -75
  764. langflow/components/pgvector/__init__.py +0 -34
  765. langflow/components/pgvector/pgvector.py +0 -72
  766. langflow/components/pinecone/__init__.py +0 -34
  767. langflow/components/pinecone/pinecone.py +0 -134
  768. langflow/components/processing/alter_metadata.py +0 -108
  769. langflow/components/processing/batch_run.py +0 -205
  770. langflow/components/processing/combine_text.py +0 -39
  771. langflow/components/processing/create_data.py +0 -110
  772. langflow/components/processing/data_operations.py +0 -438
  773. langflow/components/processing/data_to_dataframe.py +0 -70
  774. langflow/components/processing/dataframe_operations.py +0 -321
  775. langflow/components/processing/extract_key.py +0 -53
  776. langflow/components/processing/filter_data.py +0 -42
  777. langflow/components/processing/filter_data_values.py +0 -88
  778. langflow/components/processing/json_cleaner.py +0 -103
  779. langflow/components/processing/lambda_filter.py +0 -154
  780. langflow/components/processing/llm_router.py +0 -499
  781. langflow/components/processing/merge_data.py +0 -90
  782. langflow/components/processing/message_to_data.py +0 -36
  783. langflow/components/processing/parse_data.py +0 -70
  784. langflow/components/processing/parse_dataframe.py +0 -68
  785. langflow/components/processing/parse_json_data.py +0 -90
  786. langflow/components/processing/parser.py +0 -143
  787. langflow/components/processing/prompt.py +0 -67
  788. langflow/components/processing/python_repl_core.py +0 -98
  789. langflow/components/processing/regex.py +0 -82
  790. langflow/components/processing/save_file.py +0 -208
  791. langflow/components/processing/select_data.py +0 -48
  792. langflow/components/processing/split_text.py +0 -141
  793. langflow/components/processing/structured_output.py +0 -202
  794. langflow/components/processing/update_data.py +0 -160
  795. langflow/components/prototypes/__init__.py +0 -34
  796. langflow/components/prototypes/python_function.py +0 -73
  797. langflow/components/qdrant/__init__.py +0 -34
  798. langflow/components/qdrant/qdrant.py +0 -109
  799. langflow/components/redis/__init__.py +0 -37
  800. langflow/components/redis/redis.py +0 -89
  801. langflow/components/redis/redis_chat.py +0 -43
  802. langflow/components/sambanova/__init__.py +0 -32
  803. langflow/components/sambanova/sambanova.py +0 -84
  804. langflow/components/scrapegraph/__init__.py +0 -40
  805. langflow/components/scrapegraph/scrapegraph_markdownify_api.py +0 -64
  806. langflow/components/scrapegraph/scrapegraph_search_api.py +0 -64
  807. langflow/components/scrapegraph/scrapegraph_smart_scraper_api.py +0 -71
  808. langflow/components/searchapi/__init__.py +0 -36
  809. langflow/components/searchapi/search.py +0 -79
  810. langflow/components/serpapi/__init__.py +0 -3
  811. langflow/components/serpapi/serp.py +0 -115
  812. langflow/components/serper/__init__.py +0 -3
  813. langflow/components/serper/google_serper_api_core.py +0 -74
  814. langflow/components/supabase/__init__.py +0 -37
  815. langflow/components/supabase/supabase.py +0 -76
  816. langflow/components/tavily/__init__.py +0 -4
  817. langflow/components/tavily/tavily_extract.py +0 -117
  818. langflow/components/tavily/tavily_search.py +0 -212
  819. langflow/components/textsplitters/__init__.py +0 -0
  820. langflow/components/toolkits/__init__.py +0 -0
  821. langflow/components/tools/__init__.py +0 -72
  822. langflow/components/tools/calculator.py +0 -103
  823. langflow/components/tools/google_search_api.py +0 -45
  824. langflow/components/tools/google_serper_api.py +0 -115
  825. langflow/components/tools/python_code_structured_tool.py +0 -327
  826. langflow/components/tools/python_repl.py +0 -97
  827. langflow/components/tools/search_api.py +0 -87
  828. langflow/components/tools/searxng.py +0 -145
  829. langflow/components/tools/serp_api.py +0 -119
  830. langflow/components/tools/tavily_search_tool.py +0 -344
  831. langflow/components/tools/wikidata_api.py +0 -102
  832. langflow/components/tools/wikipedia_api.py +0 -49
  833. langflow/components/tools/yahoo_finance.py +0 -124
  834. langflow/components/twelvelabs/__init__.py +0 -52
  835. langflow/components/twelvelabs/convert_astra_results.py +0 -84
  836. langflow/components/twelvelabs/pegasus_index.py +0 -311
  837. langflow/components/twelvelabs/split_video.py +0 -291
  838. langflow/components/twelvelabs/text_embeddings.py +0 -57
  839. langflow/components/twelvelabs/twelvelabs_pegasus.py +0 -408
  840. langflow/components/twelvelabs/video_embeddings.py +0 -100
  841. langflow/components/twelvelabs/video_file.py +0 -179
  842. langflow/components/unstructured/__init__.py +0 -3
  843. langflow/components/unstructured/unstructured.py +0 -121
  844. langflow/components/upstash/__init__.py +0 -34
  845. langflow/components/upstash/upstash.py +0 -124
  846. langflow/components/vectara/__init__.py +0 -37
  847. langflow/components/vectara/vectara.py +0 -97
  848. langflow/components/vectara/vectara_rag.py +0 -164
  849. langflow/components/vectorstores/__init__.py +0 -34
  850. langflow/components/vectorstores/local_db.py +0 -261
  851. langflow/components/vertexai/__init__.py +0 -37
  852. langflow/components/vertexai/vertexai.py +0 -71
  853. langflow/components/vertexai/vertexai_embeddings.py +0 -67
  854. langflow/components/weaviate/__init__.py +0 -34
  855. langflow/components/weaviate/weaviate.py +0 -89
  856. langflow/components/wikipedia/__init__.py +0 -4
  857. langflow/components/wikipedia/wikidata.py +0 -86
  858. langflow/components/wikipedia/wikipedia.py +0 -53
  859. langflow/components/wolframalpha/__init__.py +0 -3
  860. langflow/components/wolframalpha/wolfram_alpha_api.py +0 -54
  861. langflow/components/xai/__init__.py +0 -32
  862. langflow/components/xai/xai.py +0 -167
  863. langflow/components/yahoosearch/__init__.py +0 -3
  864. langflow/components/yahoosearch/yahoo.py +0 -137
  865. langflow/components/youtube/__init__.py +0 -52
  866. langflow/components/youtube/channel.py +0 -227
  867. langflow/components/youtube/comments.py +0 -231
  868. langflow/components/youtube/playlist.py +0 -33
  869. langflow/components/youtube/search.py +0 -120
  870. langflow/components/youtube/trending.py +0 -285
  871. langflow/components/youtube/video_details.py +0 -263
  872. langflow/components/youtube/youtube_transcripts.py +0 -118
  873. langflow/components/zep/__init__.py +0 -3
  874. langflow/components/zep/zep.py +0 -44
  875. langflow/custom/attributes.py +0 -86
  876. langflow/custom/code_parser/__init__.py +0 -3
  877. langflow/custom/code_parser/code_parser.py +0 -361
  878. langflow/custom/custom_component/base_component.py +0 -118
  879. langflow/custom/dependency_analyzer.py +0 -165
  880. langflow/custom/directory_reader/__init__.py +0 -3
  881. langflow/custom/directory_reader/directory_reader.py +0 -359
  882. langflow/custom/directory_reader/utils.py +0 -171
  883. langflow/custom/eval.py +0 -12
  884. langflow/custom/schema.py +0 -32
  885. langflow/custom/tree_visitor.py +0 -21
  886. langflow/frontend/assets/lazyIconImports-Ci-S9xBA.js +0 -2
  887. langflow/graph/edge/__init__.py +0 -0
  888. langflow/graph/edge/base.py +0 -277
  889. langflow/graph/edge/schema.py +0 -119
  890. langflow/graph/edge/utils.py +0 -0
  891. langflow/graph/graph/__init__.py +0 -0
  892. langflow/graph/graph/ascii.py +0 -202
  893. langflow/graph/graph/base.py +0 -2185
  894. langflow/graph/graph/constants.py +0 -58
  895. langflow/graph/graph/runnable_vertices_manager.py +0 -133
  896. langflow/graph/graph/schema.py +0 -53
  897. langflow/graph/graph/state_model.py +0 -66
  898. langflow/graph/graph/utils.py +0 -1024
  899. langflow/graph/schema.py +0 -75
  900. langflow/graph/state/__init__.py +0 -0
  901. langflow/graph/state/model.py +0 -237
  902. langflow/graph/utils.py +0 -229
  903. langflow/graph/vertex/__init__.py +0 -0
  904. langflow/graph/vertex/base.py +0 -811
  905. langflow/graph/vertex/constants.py +0 -0
  906. langflow/graph/vertex/exceptions.py +0 -4
  907. langflow/graph/vertex/param_handler.py +0 -255
  908. langflow/graph/vertex/schema.py +0 -26
  909. langflow/graph/vertex/utils.py +0 -19
  910. langflow/graph/vertex/vertex_types.py +0 -489
  911. langflow/legacy_custom/__init__.py +0 -0
  912. langflow/legacy_custom/customs.py +0 -16
  913. langflow/load/load.py +0 -250
  914. langflow/logging/logger.py +0 -369
  915. langflow/processing/utils.py +0 -25
  916. langflow/schema/openai_responses_schemas.py +0 -74
  917. langflow/schema/serialize.py +0 -13
  918. langflow/services/chat/config.py +0 -2
  919. langflow/services/settings/auth.py +0 -130
  920. langflow/services/settings/constants.py +0 -31
  921. langflow/services/settings/manager.py +0 -49
  922. langflow/services/settings/utils.py +0 -40
  923. langflow/template/field/prompt.py +0 -2
  924. langflow/template/frontend_node/__init__.py +0 -6
  925. langflow/template/frontend_node/base.py +0 -212
  926. langflow/template/frontend_node/constants.py +0 -65
  927. langflow/template/frontend_node/custom_components.py +0 -97
  928. langflow/template/template/__init__.py +0 -0
  929. langflow/template/template/base.py +0 -99
  930. langflow/utils/async_helpers.py +0 -42
  931. langflow/utils/concurrency.py +0 -60
  932. langflow/utils/util_strings.py +0 -56
  933. langflow_base_nightly-0.5.1.dev2.dist-info/RECORD +0 -1159
  934. {langflow_base_nightly-0.5.1.dev2.dist-info → langflow_base_nightly-0.5.1.dev4.dist-info}/WHEEL +0 -0
  935. {langflow_base_nightly-0.5.1.dev2.dist-info → langflow_base_nightly-0.5.1.dev4.dist-info}/entry_points.txt +0 -0
@@ -1,2185 +0,0 @@
1
- from __future__ import annotations
2
-
3
- import asyncio
4
- import contextlib
5
- import contextvars
6
- import copy
7
- import json
8
- import queue
9
- import threading
10
- import traceback
11
- import uuid
12
- from collections import defaultdict, deque
13
- from datetime import datetime, timezone
14
- from functools import partial
15
- from itertools import chain
16
- from typing import TYPE_CHECKING, Any, cast
17
-
18
- from langflow.exceptions.component import ComponentBuildError
19
- from langflow.graph.edge.base import CycleEdge, Edge
20
- from langflow.graph.graph.constants import Finish, lazy_load_vertex_dict
21
- from langflow.graph.graph.runnable_vertices_manager import RunnableVerticesManager
22
- from langflow.graph.graph.schema import GraphData, GraphDump, StartConfigDict, VertexBuildResult
23
- from langflow.graph.graph.state_model import create_state_model_from_graph
24
- from langflow.graph.graph.utils import (
25
- find_all_cycle_edges,
26
- find_cycle_vertices,
27
- find_start_component_id,
28
- get_sorted_vertices,
29
- process_flow,
30
- should_continue,
31
- )
32
- from langflow.graph.schema import InterfaceComponentTypes, RunOutputs
33
- from langflow.graph.utils import log_vertex_build
34
- from langflow.graph.vertex.base import Vertex, VertexStates
35
- from langflow.graph.vertex.schema import NodeData, NodeTypeEnum
36
- from langflow.graph.vertex.vertex_types import ComponentVertex, InterfaceVertex, StateVertex
37
- from langflow.logging.logger import LogConfig, configure, logger
38
- from langflow.schema.dotdict import dotdict
39
- from langflow.schema.schema import INPUT_FIELD_NAME, InputType, OutputValue
40
- from langflow.services.cache.utils import CacheMiss
41
- from langflow.services.deps import get_chat_service, get_tracing_service
42
- from langflow.utils.async_helpers import run_until_complete
43
-
44
- if TYPE_CHECKING:
45
- from collections.abc import Callable, Generator, Iterable
46
-
47
- from langflow.api.v1.schemas import InputValueRequest
48
- from langflow.custom.custom_component.component import Component
49
- from langflow.events.event_manager import EventManager
50
- from langflow.graph.edge.schema import EdgeData
51
- from langflow.graph.schema import ResultData
52
- from langflow.services.chat.schema import GetCache, SetCache
53
- from langflow.services.tracing.service import TracingService
54
-
55
-
56
- class Graph:
57
- """A class representing a graph of vertices and edges."""
58
-
59
- def __init__(
60
- self,
61
- start: Component | None = None,
62
- end: Component | None = None,
63
- flow_id: str | None = None,
64
- flow_name: str | None = None,
65
- description: str | None = None,
66
- user_id: str | None = None,
67
- log_config: LogConfig | None = None,
68
- context: dict[str, Any] | None = None,
69
- ) -> None:
70
- """Initializes a new Graph instance.
71
-
72
- If both start and end components are provided, the graph is initialized and prepared for execution.
73
- If only one is provided, a ValueError is raised. The context must be a dictionary if specified,
74
- otherwise a TypeError is raised. Internal data structures for vertices, edges, state management,
75
- run management, and tracing are set up during initialization.
76
- """
77
- if log_config:
78
- configure(**log_config)
79
-
80
- self._start = start
81
- self._state_model = None
82
- self._end = end
83
- self._prepared = False
84
- self._runs = 0
85
- self._updates = 0
86
- self.flow_id = flow_id
87
- self.flow_name = flow_name
88
- self.description = description
89
- self.user_id = user_id
90
- self._is_input_vertices: list[str] = []
91
- self._is_output_vertices: list[str] = []
92
- self._is_state_vertices: list[str] | None = None
93
- self.has_session_id_vertices: list[str] = []
94
- self._sorted_vertices_layers: list[list[str]] = []
95
- self._run_id = ""
96
- self._session_id = ""
97
- self._start_time = datetime.now(timezone.utc)
98
- self.inactivated_vertices: set = set()
99
- self.activated_vertices: list[str] = []
100
- self.vertices_layers: list[list[str]] = []
101
- self.vertices_to_run: set[str] = set()
102
- self.stop_vertex: str | None = None
103
- self.inactive_vertices: set = set()
104
- self.edges: list[CycleEdge] = []
105
- self.vertices: list[Vertex] = []
106
- self.run_manager = RunnableVerticesManager()
107
- self._vertices: list[NodeData] = []
108
- self._edges: list[EdgeData] = []
109
-
110
- self.top_level_vertices: list[str] = []
111
- self.vertex_map: dict[str, Vertex] = {}
112
- self.predecessor_map: dict[str, list[str]] = defaultdict(list)
113
- self.successor_map: dict[str, list[str]] = defaultdict(list)
114
- self.in_degree_map: dict[str, int] = defaultdict(int)
115
- self.parent_child_map: dict[str, list[str]] = defaultdict(list)
116
- self._run_queue: deque[str] = deque()
117
- self._first_layer: list[str] = []
118
- self._lock = asyncio.Lock()
119
- self.raw_graph_data: GraphData = {"nodes": [], "edges": []}
120
- self._is_cyclic: bool | None = None
121
- self._cycles: list[tuple[str, str]] | None = None
122
- self._cycle_vertices: set[str] | None = None
123
- self._call_order: list[str] = []
124
- self._snapshots: list[dict[str, Any]] = []
125
- self._end_trace_tasks: set[asyncio.Task] = set()
126
-
127
- if context and not isinstance(context, dict):
128
- msg = "Context must be a dictionary"
129
- raise TypeError(msg)
130
- self._context = dotdict(context or {})
131
- try:
132
- self.tracing_service: TracingService | None = get_tracing_service()
133
- except Exception: # noqa: BLE001
134
- logger.exception("Error getting tracing service")
135
- self.tracing_service = None
136
- if start is not None and end is not None:
137
- self._set_start_and_end(start, end)
138
- self.prepare(start_component_id=start._id)
139
- if (start is not None and end is None) or (start is None and end is not None):
140
- msg = "You must provide both input and output components"
141
- raise ValueError(msg)
142
-
143
- @property
144
- def context(self) -> dotdict:
145
- if isinstance(self._context, dotdict):
146
- return self._context
147
- return dotdict(self._context)
148
-
149
- @context.setter
150
- def context(self, value: dict[str, Any]):
151
- if not isinstance(value, dict):
152
- msg = "Context must be a dictionary"
153
- raise TypeError(msg)
154
- if isinstance(value, dict):
155
- value = dotdict(value)
156
- self._context = value
157
-
158
- @property
159
- def session_id(self):
160
- return self._session_id
161
-
162
- @session_id.setter
163
- def session_id(self, value: str):
164
- self._session_id = value
165
-
166
- @property
167
- def state_model(self):
168
- if not self._state_model:
169
- self._state_model = create_state_model_from_graph(self)
170
- return self._state_model
171
-
172
- def __add__(self, other):
173
- if not isinstance(other, Graph):
174
- msg = "Can only add Graph objects"
175
- raise TypeError(msg)
176
- # Add the vertices and edges from the other graph to this graph
177
- new_instance = copy.deepcopy(self)
178
- for vertex in other.vertices:
179
- # This updates the edges as well
180
- new_instance.add_vertex(vertex)
181
- new_instance.build_graph_maps(new_instance.edges)
182
- new_instance.define_vertices_lists()
183
- return new_instance
184
-
185
- def __iadd__(self, other):
186
- if not isinstance(other, Graph):
187
- msg = "Can only add Graph objects"
188
- raise TypeError(msg)
189
- # Add the vertices and edges from the other graph to this graph
190
- for vertex in other.vertices:
191
- # This updates the edges as well
192
- self.add_vertex(vertex)
193
- self.build_graph_maps(self.edges)
194
- self.define_vertices_lists()
195
- return self
196
-
197
- def dumps(
198
- self,
199
- name: str | None = None,
200
- description: str | None = None,
201
- endpoint_name: str | None = None,
202
- ) -> str:
203
- graph_dict = self.dump(name, description, endpoint_name)
204
- return json.dumps(graph_dict, indent=4, sort_keys=True)
205
-
206
- def dump(
207
- self, name: str | None = None, description: str | None = None, endpoint_name: str | None = None
208
- ) -> GraphDump:
209
- if self.raw_graph_data != {"nodes": [], "edges": []}:
210
- data_dict = self.raw_graph_data
211
- else:
212
- # we need to convert the vertices and edges to json
213
- nodes = [node.to_data() for node in self.vertices]
214
- edges = [edge.to_data() for edge in self.edges]
215
- self.raw_graph_data = {"nodes": nodes, "edges": edges}
216
- data_dict = self.raw_graph_data
217
- graph_dict: GraphDump = {
218
- "data": data_dict,
219
- "is_component": len(data_dict.get("nodes", [])) == 1 and data_dict["edges"] == [],
220
- }
221
- if name:
222
- graph_dict["name"] = name
223
- elif name is None and self.flow_name:
224
- graph_dict["name"] = self.flow_name
225
- if description:
226
- graph_dict["description"] = description
227
- elif description is None and self.description:
228
- graph_dict["description"] = self.description
229
- graph_dict["endpoint_name"] = str(endpoint_name)
230
- return graph_dict
231
-
232
- def add_nodes_and_edges(self, nodes: list[NodeData], edges: list[EdgeData]) -> None:
233
- self._vertices = nodes
234
- self._edges = edges
235
- self.raw_graph_data = {"nodes": nodes, "edges": edges}
236
- self.top_level_vertices = []
237
- for vertex in self._vertices:
238
- if vertex_id := vertex.get("id"):
239
- self.top_level_vertices.append(vertex_id)
240
- if vertex_id in self.cycle_vertices:
241
- self.run_manager.add_to_cycle_vertices(vertex_id)
242
- self._graph_data = process_flow(self.raw_graph_data)
243
-
244
- self._vertices = self._graph_data["nodes"]
245
- self._edges = self._graph_data["edges"]
246
- self.initialize()
247
-
248
- def add_component(self, component: Component, component_id: str | None = None) -> str:
249
- component_id = component_id or component._id
250
- if component_id in self.vertex_map:
251
- return component_id
252
- component._id = component_id
253
- if component_id in self.vertex_map:
254
- msg = f"Component ID {component_id} already exists"
255
- raise ValueError(msg)
256
- frontend_node = component.to_frontend_node()
257
- self._vertices.append(frontend_node)
258
- vertex = self._create_vertex(frontend_node)
259
- vertex.add_component_instance(component)
260
- self._add_vertex(vertex)
261
- if component._edges:
262
- for edge in component._edges:
263
- self._add_edge(edge)
264
-
265
- if component._components:
266
- for _component in component._components:
267
- self.add_component(_component)
268
-
269
- return component_id
270
-
271
- def _set_start_and_end(self, start: Component, end: Component) -> None:
272
- if not hasattr(start, "to_frontend_node"):
273
- msg = f"start must be a Component. Got {type(start)}"
274
- raise TypeError(msg)
275
- if not hasattr(end, "to_frontend_node"):
276
- msg = f"end must be a Component. Got {type(end)}"
277
- raise TypeError(msg)
278
- self.add_component(start, start._id)
279
- self.add_component(end, end._id)
280
-
281
- def add_component_edge(self, source_id: str, output_input_tuple: tuple[str, str], target_id: str) -> None:
282
- source_vertex = self.get_vertex(source_id)
283
- if not isinstance(source_vertex, ComponentVertex):
284
- msg = f"Source vertex {source_id} is not a component vertex."
285
- raise TypeError(msg)
286
- target_vertex = self.get_vertex(target_id)
287
- if not isinstance(target_vertex, ComponentVertex):
288
- msg = f"Target vertex {target_id} is not a component vertex."
289
- raise TypeError(msg)
290
- output_name, input_name = output_input_tuple
291
- if source_vertex.custom_component is None:
292
- msg = f"Source vertex {source_id} does not have a custom component."
293
- raise ValueError(msg)
294
- if target_vertex.custom_component is None:
295
- msg = f"Target vertex {target_id} does not have a custom component."
296
- raise ValueError(msg)
297
-
298
- try:
299
- input_field = target_vertex.get_input(input_name)
300
- input_types = input_field.input_types
301
- input_field_type = str(input_field.field_type)
302
- except ValueError as e:
303
- input_field = target_vertex.data.get("node", {}).get("template", {}).get(input_name)
304
- if not input_field:
305
- msg = f"Input field {input_name} not found in target vertex {target_id}"
306
- raise ValueError(msg) from e
307
- input_types = input_field.get("input_types", [])
308
- input_field_type = input_field.get("type", "")
309
-
310
- edge_data: EdgeData = {
311
- "source": source_id,
312
- "target": target_id,
313
- "data": {
314
- "sourceHandle": {
315
- "dataType": source_vertex.custom_component.name
316
- or source_vertex.custom_component.__class__.__name__,
317
- "id": source_vertex.id,
318
- "name": output_name,
319
- "output_types": source_vertex.get_output(output_name).types,
320
- },
321
- "targetHandle": {
322
- "fieldName": input_name,
323
- "id": target_vertex.id,
324
- "inputTypes": input_types,
325
- "type": input_field_type,
326
- },
327
- },
328
- }
329
- self._add_edge(edge_data)
330
-
331
- async def async_start(
332
- self,
333
- inputs: list[dict] | None = None,
334
- max_iterations: int | None = None,
335
- config: StartConfigDict | None = None,
336
- event_manager: EventManager | None = None,
337
- ):
338
- if not self._prepared:
339
- msg = "Graph not prepared. Call prepare() first."
340
- raise ValueError(msg)
341
- # The idea is for this to return a generator that yields the result of
342
- # each step call and raise StopIteration when the graph is done
343
- if config is not None:
344
- self.__apply_config(config)
345
- for _input in inputs or []:
346
- for key, value in _input.items():
347
- vertex = self.get_vertex(key)
348
- vertex.set_input_value(key, value)
349
- # I want to keep a counter of how many times result.vertex.id
350
- # has been yielded
351
- yielded_counts: dict[str, int] = defaultdict(int)
352
-
353
- while should_continue(yielded_counts, max_iterations):
354
- result = await self.astep(event_manager=event_manager)
355
- yield result
356
- if hasattr(result, "vertex"):
357
- yielded_counts[result.vertex.id] += 1
358
- if isinstance(result, Finish):
359
- return
360
-
361
- msg = "Max iterations reached"
362
- raise ValueError(msg)
363
-
364
- def _snapshot(self):
365
- return {
366
- "_run_queue": self._run_queue.copy(),
367
- "_first_layer": self._first_layer.copy(),
368
- "vertices_layers": copy.deepcopy(self.vertices_layers),
369
- "vertices_to_run": copy.deepcopy(self.vertices_to_run),
370
- "run_manager": copy.deepcopy(self.run_manager.to_dict()),
371
- }
372
-
373
- def __apply_config(self, config: StartConfigDict) -> None:
374
- for vertex in self.vertices:
375
- if vertex.custom_component is None:
376
- continue
377
- for output in vertex.custom_component._outputs_map.values():
378
- for key, value in config["output"].items():
379
- setattr(output, key, value)
380
-
381
- def start(
382
- self,
383
- inputs: list[dict] | None = None,
384
- max_iterations: int | None = None,
385
- config: StartConfigDict | None = None,
386
- event_manager: EventManager | None = None,
387
- ) -> Generator:
388
- """Starts the graph execution synchronously by creating a new event loop in a separate thread.
389
-
390
- Args:
391
- inputs: Optional list of input dictionaries
392
- max_iterations: Optional maximum number of iterations
393
- config: Optional configuration dictionary
394
- event_manager: Optional event manager
395
-
396
- Returns:
397
- Generator yielding results from graph execution
398
- """
399
- if self.is_cyclic and max_iterations is None:
400
- msg = "You must specify a max_iterations if the graph is cyclic"
401
- raise ValueError(msg)
402
-
403
- if config is not None:
404
- self.__apply_config(config)
405
-
406
- # Create a queue for passing results and errors between threads
407
- result_queue: queue.Queue[VertexBuildResult | Exception | None] = queue.Queue()
408
-
409
- # Function to run async code in separate thread
410
- def run_async_code():
411
- # Create new event loop for this thread
412
- loop = asyncio.new_event_loop()
413
- asyncio.set_event_loop(loop)
414
-
415
- try:
416
- # Run the async generator
417
- async_gen = self.async_start(inputs, max_iterations, event_manager)
418
-
419
- while True:
420
- try:
421
- # Get next result from async generator
422
- result = loop.run_until_complete(anext(async_gen))
423
- result_queue.put(result)
424
-
425
- if isinstance(result, Finish):
426
- break
427
-
428
- except StopAsyncIteration:
429
- break
430
- except ValueError as e:
431
- # Put the exception in the queue
432
- result_queue.put(e)
433
- break
434
-
435
- finally:
436
- # Ensure all pending tasks are completed
437
- pending = asyncio.all_tasks(loop)
438
- if pending:
439
- # Create a future to gather all pending tasks
440
- cleanup_future = asyncio.gather(*pending, return_exceptions=True)
441
- loop.run_until_complete(cleanup_future)
442
-
443
- # Close the loop
444
- loop.close()
445
- # Signal completion
446
- result_queue.put(None)
447
-
448
- # Start thread for async execution
449
- thread = threading.Thread(target=run_async_code)
450
- thread.start()
451
-
452
- # Yield results from queue
453
- while True:
454
- result = result_queue.get()
455
- if result is None:
456
- break
457
- if isinstance(result, Exception):
458
- raise result
459
- yield result
460
-
461
- # Wait for thread to complete
462
- thread.join()
463
-
464
- def _add_edge(self, edge: EdgeData) -> None:
465
- self.add_edge(edge)
466
- source_id = edge["data"]["sourceHandle"]["id"]
467
- target_id = edge["data"]["targetHandle"]["id"]
468
- self.predecessor_map[target_id].append(source_id)
469
- self.successor_map[source_id].append(target_id)
470
- self.in_degree_map[target_id] += 1
471
- self.parent_child_map[source_id].append(target_id)
472
-
473
- def add_node(self, node: NodeData) -> None:
474
- self._vertices.append(node)
475
-
476
- def add_edge(self, edge: EdgeData) -> None:
477
- # Check if the edge already exists
478
- if edge in self._edges:
479
- return
480
- self._edges.append(edge)
481
-
482
- def initialize(self) -> None:
483
- self._build_graph()
484
- self.build_graph_maps(self.edges)
485
- self.define_vertices_lists()
486
-
487
- @property
488
- def is_state_vertices(self) -> list[str]:
489
- """Returns a cached list of vertex IDs for vertices marked as state vertices.
490
-
491
- The list is computed on first access by filtering vertices with `is_state` set to True and is
492
- cached for future calls.
493
- """
494
- if self._is_state_vertices is None:
495
- self._is_state_vertices = [vertex.id for vertex in self.vertices if vertex.is_state]
496
- return self._is_state_vertices
497
-
498
- def activate_state_vertices(self, name: str, caller: str) -> None:
499
- """Activates vertices associated with a given state name.
500
-
501
- Marks vertices with the specified state name, as well as their successors and related
502
- predecessors. The state manager is then updated with the new state record.
503
- """
504
- vertices_ids = set()
505
- new_predecessor_map = {}
506
- activated_vertices = []
507
- for vertex_id in self.is_state_vertices:
508
- caller_vertex = self.get_vertex(caller)
509
- vertex = self.get_vertex(vertex_id)
510
- if vertex_id == caller or vertex.display_name == caller_vertex.display_name:
511
- continue
512
- ctx_key = vertex.raw_params.get("context_key")
513
- if isinstance(ctx_key, str) and name in ctx_key and vertex_id != caller and isinstance(vertex, StateVertex):
514
- activated_vertices.append(vertex_id)
515
- vertices_ids.add(vertex_id)
516
- successors = self.get_all_successors(vertex, flat=True)
517
- # Update run_manager.run_predecessors because we are activating vertices
518
- # The run_prdecessors is the predecessor map of the vertices
519
- # we remove the vertex_id from the predecessor map whenever we run a vertex
520
- # So we need to get all edges of the vertex and successors
521
- # and run self.build_adjacency_maps(edges) to get the new predecessor map
522
- # that is not complete but we can use to update the run_predecessors
523
- successors_predecessors = set()
524
- for sucessor in successors:
525
- successors_predecessors.update(self.get_all_predecessors(sucessor))
526
-
527
- edges_set = set()
528
- for _vertex in [vertex, *successors, *successors_predecessors]:
529
- edges_set.update(_vertex.edges)
530
- if _vertex.state == VertexStates.INACTIVE:
531
- _vertex.set_state("ACTIVE")
532
-
533
- vertices_ids.add(_vertex.id)
534
- edges = list(edges_set)
535
- predecessor_map, _ = self.build_adjacency_maps(edges)
536
- new_predecessor_map.update(predecessor_map)
537
-
538
- vertices_ids.update(new_predecessor_map.keys())
539
- vertices_ids.update(v_id for value_list in new_predecessor_map.values() for v_id in value_list)
540
-
541
- self.activated_vertices = activated_vertices
542
- self.vertices_to_run.update(vertices_ids)
543
- self.run_manager.update_run_state(
544
- run_predecessors=new_predecessor_map,
545
- vertices_to_run=self.vertices_to_run,
546
- )
547
-
548
- def reset_activated_vertices(self) -> None:
549
- """Resets the activated vertices in the graph."""
550
- self.activated_vertices = []
551
-
552
- def validate_stream(self) -> None:
553
- """Validates the stream configuration of the graph.
554
-
555
- If there are two vertices in the same graph (connected by edges)
556
- that have `stream=True` or `streaming=True`, raises a `ValueError`.
557
-
558
- Raises:
559
- ValueError: If two connected vertices have `stream=True` or `streaming=True`.
560
- """
561
- for vertex in self.vertices:
562
- if vertex.params.get("stream") or vertex.params.get("streaming"):
563
- successors = self.get_all_successors(vertex)
564
- for successor in successors:
565
- if successor.params.get("stream") or successor.params.get("streaming"):
566
- msg = (
567
- f"Components {vertex.display_name} and {successor.display_name} "
568
- "are connected and both have stream or streaming set to True"
569
- )
570
- raise ValueError(msg)
571
-
572
- @property
573
- def first_layer(self):
574
- if self._first_layer is None:
575
- msg = "Graph not prepared. Call prepare() first."
576
- raise ValueError(msg)
577
- return self._first_layer
578
-
579
- @property
580
- def is_cyclic(self):
581
- """Check if the graph has any cycles.
582
-
583
- Returns:
584
- bool: True if the graph has any cycles, False otherwise.
585
- """
586
- if self._is_cyclic is None:
587
- self._is_cyclic = bool(self.cycle_vertices)
588
- return self._is_cyclic
589
-
590
- @property
591
- def run_id(self):
592
- """The ID of the current run.
593
-
594
- Returns:
595
- str: The run ID.
596
-
597
- Raises:
598
- ValueError: If the run ID is not set.
599
- """
600
- if not self._run_id:
601
- msg = "Run ID not set"
602
- raise ValueError(msg)
603
- return self._run_id
604
-
605
- def set_run_id(self, run_id: uuid.UUID | str | None = None) -> None:
606
- """Sets the ID of the current run.
607
-
608
- Args:
609
- run_id (str): The run ID.
610
- """
611
- if run_id is None:
612
- run_id = uuid.uuid4()
613
-
614
- self._run_id = str(run_id)
615
-
616
- async def initialize_run(self) -> None:
617
- if not self._run_id:
618
- self.set_run_id()
619
- if self.tracing_service:
620
- run_name = f"{self.flow_name} - {self.flow_id}"
621
- await self.tracing_service.start_tracers(
622
- run_id=uuid.UUID(self._run_id),
623
- run_name=run_name,
624
- user_id=self.user_id,
625
- session_id=self.session_id,
626
- )
627
-
628
- def _end_all_traces_async(self, outputs: dict[str, Any] | None = None, error: Exception | None = None) -> None:
629
- task = asyncio.create_task(self.end_all_traces(outputs, error))
630
- self._end_trace_tasks.add(task)
631
- task.add_done_callback(self._end_trace_tasks.discard)
632
-
633
- def end_all_traces_in_context(
634
- self,
635
- outputs: dict[str, Any] | None = None,
636
- error: Exception | None = None,
637
- ) -> Callable:
638
- # BackgroundTasks run in different context, so we need to copy the context
639
- context = contextvars.copy_context()
640
-
641
- async def async_end_traces_func():
642
- await asyncio.create_task(self.end_all_traces(outputs, error), context=context)
643
-
644
- return async_end_traces_func
645
-
646
- async def end_all_traces(self, outputs: dict[str, Any] | None = None, error: Exception | None = None) -> None:
647
- if not self.tracing_service:
648
- return
649
- self._end_time = datetime.now(timezone.utc)
650
- if outputs is None:
651
- outputs = {}
652
- outputs |= self.metadata
653
- await self.tracing_service.end_tracers(outputs, error)
654
-
655
- @property
656
- def sorted_vertices_layers(self) -> list[list[str]]:
657
- """Returns the sorted layers of vertex IDs by type.
658
-
659
- Each layer in the returned list contains vertex IDs grouped by their classification,
660
- such as input, output, session, or state vertices. Sorting is performed if not already done.
661
- """
662
- if not self._sorted_vertices_layers:
663
- self.sort_vertices()
664
- return self._sorted_vertices_layers
665
-
666
- def define_vertices_lists(self) -> None:
667
- """Populates internal lists of input, output, session ID, and state vertex IDs.
668
-
669
- Iterates over all vertices and appends their IDs to the corresponding internal lists
670
- based on their classification.
671
- """
672
- for vertex in self.vertices:
673
- if vertex.is_input:
674
- self._is_input_vertices.append(vertex.id)
675
- if vertex.is_output:
676
- self._is_output_vertices.append(vertex.id)
677
- if vertex.has_session_id:
678
- self.has_session_id_vertices.append(vertex.id)
679
- if vertex.is_state:
680
- if self._is_state_vertices is None:
681
- self._is_state_vertices = []
682
- self._is_state_vertices.append(vertex.id)
683
-
684
- def _set_inputs(self, input_components: list[str], inputs: dict[str, str], input_type: InputType | None) -> None:
685
- """Updates input vertices' parameters with the provided inputs, filtering by component list and input type.
686
-
687
- Only vertices whose IDs or display names match the specified input components and whose IDs contain
688
- the input type (unless input type is 'any' or None) are updated. Raises a ValueError if a specified
689
- vertex is not found.
690
- """
691
- for vertex_id in self._is_input_vertices:
692
- vertex = self.get_vertex(vertex_id)
693
- # If the vertex is not in the input_components list
694
- if input_components and (vertex_id not in input_components and vertex.display_name not in input_components):
695
- continue
696
- # If the input_type is not any and the input_type is not in the vertex id
697
- # Example: input_type = "chat" and vertex.id = "OpenAI-19ddn"
698
- if input_type is not None and input_type != "any" and input_type not in vertex.id.lower():
699
- continue
700
- if vertex is None:
701
- msg = f"Vertex {vertex_id} not found"
702
- raise ValueError(msg)
703
- vertex.update_raw_params(inputs, overwrite=True)
704
-
705
- async def _run(
706
- self,
707
- *,
708
- inputs: dict[str, str],
709
- input_components: list[str],
710
- input_type: InputType | None,
711
- outputs: list[str],
712
- stream: bool,
713
- session_id: str,
714
- fallback_to_env_vars: bool,
715
- event_manager: EventManager | None = None,
716
- ) -> list[ResultData | None]:
717
- """Runs the graph with the given inputs.
718
-
719
- Args:
720
- inputs (Dict[str, str]): The input values for the graph.
721
- input_components (list[str]): The components to run for the inputs.
722
- input_type: (Optional[InputType]): The input type.
723
- outputs (list[str]): The outputs to retrieve from the graph.
724
- stream (bool): Whether to stream the results or not.
725
- session_id (str): The session ID for the graph.
726
- fallback_to_env_vars (bool): Whether to fallback to environment variables.
727
- event_manager (EventManager | None): The event manager for the graph.
728
-
729
- Returns:
730
- List[Optional["ResultData"]]: The outputs of the graph.
731
- """
732
- if input_components and not isinstance(input_components, list):
733
- msg = f"Invalid components value: {input_components}. Expected list"
734
- raise ValueError(msg)
735
- if input_components is None:
736
- input_components = []
737
-
738
- if not isinstance(inputs.get(INPUT_FIELD_NAME, ""), str):
739
- msg = f"Invalid input value: {inputs.get(INPUT_FIELD_NAME)}. Expected string"
740
- raise TypeError(msg)
741
- if inputs:
742
- self._set_inputs(input_components, inputs, input_type)
743
- # Update all the vertices with the session_id
744
- for vertex_id in self.has_session_id_vertices:
745
- vertex = self.get_vertex(vertex_id)
746
- if vertex is None:
747
- msg = f"Vertex {vertex_id} not found"
748
- raise ValueError(msg)
749
- vertex.update_raw_params({"session_id": session_id})
750
- # Process the graph
751
- try:
752
- cache_service = get_chat_service()
753
- if self.flow_id:
754
- await cache_service.set_cache(self.flow_id, self)
755
- except Exception: # noqa: BLE001
756
- logger.exception("Error setting cache")
757
-
758
- try:
759
- # Prioritize the webhook component if it exists
760
- start_component_id = find_start_component_id(self._is_input_vertices)
761
- await self.process(
762
- start_component_id=start_component_id,
763
- fallback_to_env_vars=fallback_to_env_vars,
764
- event_manager=event_manager,
765
- )
766
- self.increment_run_count()
767
- except Exception as exc:
768
- self._end_all_traces_async(error=exc)
769
- msg = f"Error running graph: {exc}"
770
- raise ValueError(msg) from exc
771
-
772
- self._end_all_traces_async()
773
- # Get the outputs
774
- vertex_outputs = []
775
- for vertex in self.vertices:
776
- if not vertex.built:
777
- continue
778
- if vertex is None:
779
- msg = f"Vertex {vertex_id} not found"
780
- raise ValueError(msg)
781
-
782
- if not vertex.result and not stream and hasattr(vertex, "consume_async_generator"):
783
- await vertex.consume_async_generator()
784
- if (not outputs and vertex.is_output) or (vertex.display_name in outputs or vertex.id in outputs):
785
- vertex_outputs.append(vertex.result)
786
-
787
- return vertex_outputs
788
-
789
- async def arun(
790
- self,
791
- inputs: list[dict[str, str]],
792
- *,
793
- inputs_components: list[list[str]] | None = None,
794
- types: list[InputType | None] | None = None,
795
- outputs: list[str] | None = None,
796
- session_id: str | None = None,
797
- stream: bool = False,
798
- fallback_to_env_vars: bool = False,
799
- event_manager: EventManager | None = None,
800
- ) -> list[RunOutputs]:
801
- """Runs the graph with the given inputs.
802
-
803
- Args:
804
- inputs (list[Dict[str, str]]): The input values for the graph.
805
- inputs_components (Optional[list[list[str]]], optional): Components to run for the inputs. Defaults to None.
806
- types (Optional[list[Optional[InputType]]], optional): The types of the inputs. Defaults to None.
807
- outputs (Optional[list[str]], optional): The outputs to retrieve from the graph. Defaults to None.
808
- session_id (Optional[str], optional): The session ID for the graph. Defaults to None.
809
- stream (bool, optional): Whether to stream the results or not. Defaults to False.
810
- fallback_to_env_vars (bool, optional): Whether to fallback to environment variables. Defaults to False.
811
- event_manager (EventManager | None): The event manager for the graph.
812
-
813
- Returns:
814
- List[RunOutputs]: The outputs of the graph.
815
- """
816
- # inputs is {"message": "Hello, world!"}
817
- # we need to go through self.inputs and update the self.raw_params
818
- # of the vertices that are inputs
819
- # if the value is a list, we need to run multiple times
820
- vertex_outputs = []
821
- if not isinstance(inputs, list):
822
- inputs = [inputs]
823
- elif not inputs:
824
- inputs = [{}]
825
- # Length of all should be the as inputs length
826
- # just add empty lists to complete the length
827
- if inputs_components is None:
828
- inputs_components = []
829
- for _ in range(len(inputs) - len(inputs_components)):
830
- inputs_components.append([])
831
- if types is None:
832
- types = []
833
- if session_id:
834
- self.session_id = session_id
835
- for _ in range(len(inputs) - len(types)):
836
- types.append("chat") # default to chat
837
- for run_inputs, components, input_type in zip(inputs, inputs_components, types, strict=True):
838
- run_outputs = await self._run(
839
- inputs=run_inputs,
840
- input_components=components,
841
- input_type=input_type,
842
- outputs=outputs or [],
843
- stream=stream,
844
- session_id=session_id or "",
845
- fallback_to_env_vars=fallback_to_env_vars,
846
- event_manager=event_manager,
847
- )
848
- run_output_object = RunOutputs(inputs=run_inputs, outputs=run_outputs)
849
- await logger.adebug(f"Run outputs: {run_output_object}")
850
- vertex_outputs.append(run_output_object)
851
- return vertex_outputs
852
-
853
- def next_vertex_to_build(self):
854
- """Returns the next vertex to be built.
855
-
856
- Yields:
857
- str: The ID of the next vertex to be built.
858
- """
859
- yield from chain.from_iterable(self.vertices_layers)
860
-
861
- @property
862
- def metadata(self):
863
- """The metadata of the graph.
864
-
865
- Returns:
866
- dict: The metadata of the graph.
867
- """
868
- time_format = "%Y-%m-%d %H:%M:%S %Z"
869
- return {
870
- "start_time": self._start_time.strftime(time_format),
871
- "end_time": self._end_time.strftime(time_format),
872
- "time_elapsed": f"{(self._end_time - self._start_time).total_seconds()} seconds",
873
- "flow_id": self.flow_id,
874
- "flow_name": self.flow_name,
875
- }
876
-
877
- def build_graph_maps(self, edges: list[CycleEdge] | None = None, vertices: list[Vertex] | None = None) -> None:
878
- """Builds the adjacency maps for the graph."""
879
- if edges is None:
880
- edges = self.edges
881
-
882
- if vertices is None:
883
- vertices = self.vertices
884
-
885
- self.predecessor_map, self.successor_map = self.build_adjacency_maps(edges)
886
-
887
- self.in_degree_map = self.build_in_degree(edges)
888
- self.parent_child_map = self.build_parent_child_map(vertices)
889
-
890
- def reset_inactivated_vertices(self) -> None:
891
- """Resets the inactivated vertices in the graph."""
892
- for vertex_id in self.inactivated_vertices.copy():
893
- self.mark_vertex(vertex_id, "ACTIVE")
894
- self.inactivated_vertices = set()
895
- self.inactivated_vertices = set()
896
-
897
- def mark_all_vertices(self, state: str) -> None:
898
- """Marks all vertices in the graph."""
899
- for vertex in self.vertices:
900
- vertex.set_state(state)
901
-
902
- def mark_vertex(self, vertex_id: str, state: str) -> None:
903
- """Marks a vertex in the graph."""
904
- vertex = self.get_vertex(vertex_id)
905
- vertex.set_state(state)
906
- if state == VertexStates.INACTIVE:
907
- self.run_manager.remove_from_predecessors(vertex_id)
908
-
909
- def _mark_branch(
910
- self, vertex_id: str, state: str, visited: set | None = None, output_name: str | None = None
911
- ) -> set:
912
- """Marks a branch of the graph."""
913
- if visited is None:
914
- visited = set()
915
- else:
916
- self.mark_vertex(vertex_id, state)
917
- if vertex_id in visited:
918
- return visited
919
- visited.add(vertex_id)
920
-
921
- for child_id in self.parent_child_map[vertex_id]:
922
- # Only child_id that have an edge with the vertex_id through the output_name
923
- # should be marked
924
- if output_name:
925
- edge = self.get_edge(vertex_id, child_id)
926
- if edge and edge.source_handle.name != output_name:
927
- continue
928
- self._mark_branch(child_id, state, visited)
929
- return visited
930
-
931
- def mark_branch(self, vertex_id: str, state: str, output_name: str | None = None) -> None:
932
- visited = self._mark_branch(vertex_id=vertex_id, state=state, output_name=output_name)
933
- new_predecessor_map, _ = self.build_adjacency_maps(self.edges)
934
- new_predecessor_map = {k: v for k, v in new_predecessor_map.items() if k in visited}
935
- if vertex_id in self.cycle_vertices:
936
- # Remove dependencies that are not in the cycle and have run at least once
937
- new_predecessor_map = {
938
- k: [dep for dep in v if dep in self.cycle_vertices and dep in self.run_manager.ran_at_least_once]
939
- for k, v in new_predecessor_map.items()
940
- }
941
- self.run_manager.update_run_state(
942
- run_predecessors=new_predecessor_map,
943
- vertices_to_run=self.vertices_to_run,
944
- )
945
-
946
- def get_edge(self, source_id: str, target_id: str) -> CycleEdge | None:
947
- """Returns the edge between two vertices."""
948
- for edge in self.edges:
949
- if edge.source_id == source_id and edge.target_id == target_id:
950
- return edge
951
- return None
952
-
953
- def build_parent_child_map(self, vertices: list[Vertex]):
954
- parent_child_map = defaultdict(list)
955
- for vertex in vertices:
956
- parent_child_map[vertex.id] = [child.id for child in self.get_successors(vertex)]
957
- return parent_child_map
958
-
959
- def increment_run_count(self) -> None:
960
- self._runs += 1
961
-
962
- def increment_update_count(self) -> None:
963
- self._updates += 1
964
-
965
- def __getstate__(self):
966
- # Get all attributes that are useful in runs.
967
- # We don't need to save the state_manager because it is
968
- # a singleton and it is not necessary to save it
969
- return {
970
- "vertices": self.vertices,
971
- "edges": self.edges,
972
- "flow_id": self.flow_id,
973
- "flow_name": self.flow_name,
974
- "description": self.description,
975
- "user_id": self.user_id,
976
- "raw_graph_data": self.raw_graph_data,
977
- "top_level_vertices": self.top_level_vertices,
978
- "inactivated_vertices": self.inactivated_vertices,
979
- "run_manager": self.run_manager.to_dict(),
980
- "_run_id": self._run_id,
981
- "in_degree_map": self.in_degree_map,
982
- "parent_child_map": self.parent_child_map,
983
- "predecessor_map": self.predecessor_map,
984
- "successor_map": self.successor_map,
985
- "activated_vertices": self.activated_vertices,
986
- "vertices_layers": self.vertices_layers,
987
- "vertices_to_run": self.vertices_to_run,
988
- "stop_vertex": self.stop_vertex,
989
- "_run_queue": self._run_queue,
990
- "_first_layer": self._first_layer,
991
- "_vertices": self._vertices,
992
- "_edges": self._edges,
993
- "_is_input_vertices": self._is_input_vertices,
994
- "_is_output_vertices": self._is_output_vertices,
995
- "has_session_id_vertices": self.has_session_id_vertices,
996
- "_sorted_vertices_layers": self._sorted_vertices_layers,
997
- }
998
-
999
- def __deepcopy__(self, memo):
1000
- # Check if we've already copied this instance
1001
- if id(self) in memo:
1002
- return memo[id(self)]
1003
-
1004
- if self._start is not None and self._end is not None:
1005
- # Deep copy start and end components
1006
- start_copy = copy.deepcopy(self._start, memo)
1007
- end_copy = copy.deepcopy(self._end, memo)
1008
- new_graph = type(self)(
1009
- start_copy,
1010
- end_copy,
1011
- copy.deepcopy(self.flow_id, memo),
1012
- copy.deepcopy(self.flow_name, memo),
1013
- copy.deepcopy(self.user_id, memo),
1014
- )
1015
- else:
1016
- # Create a new graph without start and end, but copy flow_id, flow_name, and user_id
1017
- new_graph = type(self)(
1018
- None,
1019
- None,
1020
- copy.deepcopy(self.flow_id, memo),
1021
- copy.deepcopy(self.flow_name, memo),
1022
- copy.deepcopy(self.user_id, memo),
1023
- )
1024
- # Deep copy vertices and edges
1025
- new_graph.add_nodes_and_edges(copy.deepcopy(self._vertices, memo), copy.deepcopy(self._edges, memo))
1026
-
1027
- # Store the newly created object in memo
1028
- memo[id(self)] = new_graph
1029
-
1030
- return new_graph
1031
-
1032
- def __setstate__(self, state):
1033
- run_manager = state["run_manager"]
1034
- if isinstance(run_manager, RunnableVerticesManager):
1035
- state["run_manager"] = run_manager
1036
- else:
1037
- state["run_manager"] = RunnableVerticesManager.from_dict(run_manager)
1038
- self.__dict__.update(state)
1039
- self.vertex_map = {vertex.id: vertex for vertex in self.vertices}
1040
- self.tracing_service = get_tracing_service()
1041
- self.set_run_id(self._run_id)
1042
-
1043
- @classmethod
1044
- def from_payload(
1045
- cls,
1046
- payload: dict,
1047
- flow_id: str | None = None,
1048
- flow_name: str | None = None,
1049
- user_id: str | None = None,
1050
- context: dict | None = None,
1051
- ) -> Graph:
1052
- """Creates a graph from a payload.
1053
-
1054
- Args:
1055
- payload: The payload to create the graph from.
1056
- flow_id: The ID of the flow.
1057
- flow_name: The flow name.
1058
- user_id: The user ID.
1059
- context: Optional context dictionary for request-specific data.
1060
-
1061
- Returns:
1062
- Graph: The created graph.
1063
- """
1064
- if "data" in payload:
1065
- payload = payload["data"]
1066
- try:
1067
- vertices = payload["nodes"]
1068
- edges = payload["edges"]
1069
- graph = cls(flow_id=flow_id, flow_name=flow_name, user_id=user_id, context=context)
1070
- graph.add_nodes_and_edges(vertices, edges)
1071
- except KeyError as exc:
1072
- logger.exception(exc)
1073
- if "nodes" not in payload and "edges" not in payload:
1074
- msg = f"Invalid payload. Expected keys 'nodes' and 'edges'. Found {list(payload.keys())}"
1075
- raise ValueError(msg) from exc
1076
-
1077
- msg = f"Error while creating graph from payload: {exc}"
1078
- raise ValueError(msg) from exc
1079
- else:
1080
- return graph
1081
-
1082
- def __eq__(self, /, other: object) -> bool:
1083
- if not isinstance(other, Graph):
1084
- return False
1085
- return self.__repr__() == other.__repr__()
1086
-
1087
- # update this graph with another graph by comparing the __repr__ of each vertex
1088
- # and if the __repr__ of a vertex is not the same as the other
1089
- # then update the .data of the vertex to the self
1090
- # both graphs have the same vertices and edges
1091
- # but the data of the vertices might be different
1092
-
1093
- def update_edges_from_vertex(self, other_vertex: Vertex) -> None:
1094
- """Updates the edges of a vertex in the Graph."""
1095
- new_edges = []
1096
- for edge in self.edges:
1097
- if other_vertex.id in {edge.source_id, edge.target_id}:
1098
- continue
1099
- new_edges.append(edge)
1100
- new_edges += other_vertex.edges
1101
- self.edges = new_edges
1102
-
1103
- def vertex_data_is_identical(self, vertex: Vertex, other_vertex: Vertex) -> bool:
1104
- data_is_equivalent = vertex == other_vertex
1105
- if not data_is_equivalent:
1106
- return False
1107
- return self.vertex_edges_are_identical(vertex, other_vertex)
1108
-
1109
- @staticmethod
1110
- def vertex_edges_are_identical(vertex: Vertex, other_vertex: Vertex) -> bool:
1111
- same_length = len(vertex.edges) == len(other_vertex.edges)
1112
- if not same_length:
1113
- return False
1114
- return all(edge in other_vertex.edges for edge in vertex.edges)
1115
-
1116
- def update(self, other: Graph) -> Graph:
1117
- # Existing vertices in self graph
1118
- existing_vertex_ids = {vertex.id for vertex in self.vertices}
1119
- # Vertex IDs in the other graph
1120
- other_vertex_ids = set(other.vertex_map.keys())
1121
-
1122
- # Find vertices that are in other but not in self (new vertices)
1123
- new_vertex_ids = other_vertex_ids - existing_vertex_ids
1124
-
1125
- # Find vertices that are in self but not in other (removed vertices)
1126
- removed_vertex_ids = existing_vertex_ids - other_vertex_ids
1127
-
1128
- # Remove vertices that are not in the other graph
1129
- for vertex_id in removed_vertex_ids:
1130
- with contextlib.suppress(ValueError):
1131
- self.remove_vertex(vertex_id)
1132
-
1133
- # The order here matters because adding the vertex is required
1134
- # if any of them have edges that point to any of the new vertices
1135
- # By adding them first, them adding the edges we ensure that the
1136
- # edges have valid vertices to point to
1137
-
1138
- # Add new vertices
1139
- for vertex_id in new_vertex_ids:
1140
- new_vertex = other.get_vertex(vertex_id)
1141
- self._add_vertex(new_vertex)
1142
-
1143
- # Now update the edges
1144
- for vertex_id in new_vertex_ids:
1145
- new_vertex = other.get_vertex(vertex_id)
1146
- self._update_edges(new_vertex)
1147
- # Graph is set at the end because the edges come from the graph
1148
- # and the other graph is where the new edges and vertices come from
1149
- new_vertex.graph = self
1150
-
1151
- # Update existing vertices that have changed
1152
- for vertex_id in existing_vertex_ids.intersection(other_vertex_ids):
1153
- self_vertex = self.get_vertex(vertex_id)
1154
- other_vertex = other.get_vertex(vertex_id)
1155
- # If the vertices are not identical, update the vertex
1156
- if not self.vertex_data_is_identical(self_vertex, other_vertex):
1157
- self.update_vertex_from_another(self_vertex, other_vertex)
1158
-
1159
- self.build_graph_maps()
1160
- self.define_vertices_lists()
1161
- self.increment_update_count()
1162
- return self
1163
-
1164
- def update_vertex_from_another(self, vertex: Vertex, other_vertex: Vertex) -> None:
1165
- """Updates a vertex from another vertex.
1166
-
1167
- Args:
1168
- vertex (Vertex): The vertex to be updated.
1169
- other_vertex (Vertex): The vertex to update from.
1170
- """
1171
- vertex.full_data = other_vertex.full_data
1172
- vertex.parse_data()
1173
- # Now we update the edges of the vertex
1174
- self.update_edges_from_vertex(other_vertex)
1175
- vertex.params = {}
1176
- vertex.build_params()
1177
- vertex.graph = self
1178
- # If the vertex is frozen, we don't want
1179
- # to reset the results nor the built attribute
1180
- if not vertex.frozen:
1181
- vertex.built = False
1182
- vertex.result = None
1183
- vertex.artifacts = {}
1184
- vertex.set_top_level(self.top_level_vertices)
1185
- self.reset_all_edges_of_vertex(vertex)
1186
-
1187
- def reset_all_edges_of_vertex(self, vertex: Vertex) -> None:
1188
- """Resets all the edges of a vertex."""
1189
- for edge in vertex.edges:
1190
- for vid in [edge.source_id, edge.target_id]:
1191
- if vid in self.vertex_map:
1192
- vertex_ = self.vertex_map[vid]
1193
- if not vertex_.frozen:
1194
- vertex_.build_params()
1195
-
1196
- def _add_vertex(self, vertex: Vertex) -> None:
1197
- """Adds a vertex to the graph."""
1198
- self.vertices.append(vertex)
1199
- self.vertex_map[vertex.id] = vertex
1200
-
1201
- def add_vertex(self, vertex: Vertex) -> None:
1202
- """Adds a new vertex to the graph."""
1203
- self._add_vertex(vertex)
1204
- self._update_edges(vertex)
1205
-
1206
- def _update_edges(self, vertex: Vertex) -> None:
1207
- """Updates the edges of a vertex."""
1208
- # Vertex has edges, so we need to update the edges
1209
- for edge in vertex.edges:
1210
- if edge not in self.edges and edge.source_id in self.vertex_map and edge.target_id in self.vertex_map:
1211
- self.edges.append(edge)
1212
-
1213
- def _build_graph(self) -> None:
1214
- """Builds the graph from the vertices and edges."""
1215
- self.vertices = self._build_vertices()
1216
- self.vertex_map = {vertex.id: vertex for vertex in self.vertices}
1217
- self.edges = self._build_edges()
1218
-
1219
- # This is a hack to make sure that the LLM vertex is sent to
1220
- # the toolkit vertex
1221
- self._build_vertex_params()
1222
- self._instantiate_components_in_vertices()
1223
- self._set_cache_to_vertices_in_cycle()
1224
- self._set_cache_if_listen_notify_components()
1225
- for vertex in self.vertices:
1226
- if vertex.id in self.cycle_vertices:
1227
- self.run_manager.add_to_cycle_vertices(vertex.id)
1228
-
1229
- def _get_edges_as_list_of_tuples(self) -> list[tuple[str, str]]:
1230
- """Returns the edges of the graph as a list of tuples.
1231
-
1232
- Each tuple contains the source and target handle IDs from the edge data.
1233
-
1234
- Returns:
1235
- list[tuple[str, str]]: List of (source_id, target_id) tuples representing graph edges.
1236
- """
1237
- return [(e["data"]["sourceHandle"]["id"], e["data"]["targetHandle"]["id"]) for e in self._edges]
1238
-
1239
- def _set_cache_if_listen_notify_components(self) -> None:
1240
- """Disables caching for all vertices if Listen/Notify components are present.
1241
-
1242
- If the graph contains any Listen or Notify components, caching is disabled for all vertices
1243
- by setting cache=False on their outputs. This ensures proper handling of real-time
1244
- communication between components.
1245
- """
1246
- has_listen_or_notify_component = any(
1247
- vertex.id.split("-")[0] in {"Listen", "Notify"} for vertex in self.vertices
1248
- )
1249
- if has_listen_or_notify_component:
1250
- for vertex in self.vertices:
1251
- vertex.apply_on_outputs(lambda output_object: setattr(output_object, "cache", False))
1252
-
1253
- def _set_cache_to_vertices_in_cycle(self) -> None:
1254
- """Sets the cache to the vertices in cycle."""
1255
- edges = self._get_edges_as_list_of_tuples()
1256
- cycle_vertices = set(find_cycle_vertices(edges))
1257
- for vertex in self.vertices:
1258
- if vertex.id in cycle_vertices:
1259
- vertex.apply_on_outputs(lambda output_object: setattr(output_object, "cache", False))
1260
-
1261
- def _instantiate_components_in_vertices(self) -> None:
1262
- """Instantiates the components in the vertices."""
1263
- for vertex in self.vertices:
1264
- vertex.instantiate_component(self.user_id)
1265
-
1266
- def remove_vertex(self, vertex_id: str) -> None:
1267
- """Removes a vertex from the graph."""
1268
- vertex = self.get_vertex(vertex_id)
1269
- if vertex is None:
1270
- return
1271
- self.vertices.remove(vertex)
1272
- self.vertex_map.pop(vertex_id)
1273
- self.edges = [edge for edge in self.edges if vertex_id not in {edge.source_id, edge.target_id}]
1274
-
1275
- def _build_vertex_params(self) -> None:
1276
- """Identifies and handles the LLM vertex within the graph."""
1277
- for vertex in self.vertices:
1278
- vertex.build_params()
1279
-
1280
- def _validate_vertex(self, vertex: Vertex) -> bool:
1281
- """Validates a vertex."""
1282
- # All vertices that do not have edges are invalid
1283
- return len(self.get_vertex_edges(vertex.id)) > 0
1284
-
1285
- def get_vertex(self, vertex_id: str) -> Vertex:
1286
- """Returns a vertex by id."""
1287
- try:
1288
- return self.vertex_map[vertex_id]
1289
- except KeyError as e:
1290
- msg = f"Vertex {vertex_id} not found"
1291
- raise ValueError(msg) from e
1292
-
1293
- def get_root_of_group_node(self, vertex_id: str) -> Vertex:
1294
- """Returns the root of a group node."""
1295
- if vertex_id in self.top_level_vertices:
1296
- # Get all vertices with vertex_id as .parent_node_id
1297
- # then get the one at the top
1298
- vertices = [vertex for vertex in self.vertices if vertex.parent_node_id == vertex_id]
1299
- # Now go through successors of the vertices
1300
- # and get the one that none of its successors is in vertices
1301
- for vertex in vertices:
1302
- successors = self.get_all_successors(vertex, recursive=False)
1303
- if not any(successor in vertices for successor in successors):
1304
- return vertex
1305
- msg = f"Vertex {vertex_id} is not a top level vertex or no root vertex found"
1306
- raise ValueError(msg)
1307
-
1308
- def get_next_in_queue(self):
1309
- if not self._run_queue:
1310
- return None
1311
- return self._run_queue.popleft()
1312
-
1313
- def extend_run_queue(self, vertices: list[str]) -> None:
1314
- self._run_queue.extend(vertices)
1315
-
1316
- async def astep(
1317
- self,
1318
- inputs: InputValueRequest | None = None,
1319
- files: list[str] | None = None,
1320
- user_id: str | None = None,
1321
- event_manager: EventManager | None = None,
1322
- ):
1323
- if not self._prepared:
1324
- msg = "Graph not prepared. Call prepare() first."
1325
- raise ValueError(msg)
1326
- if not self._run_queue:
1327
- self._end_all_traces_async()
1328
- return Finish()
1329
- vertex_id = self.get_next_in_queue()
1330
- chat_service = get_chat_service()
1331
- vertex_build_result = await self.build_vertex(
1332
- vertex_id=vertex_id,
1333
- user_id=user_id,
1334
- inputs_dict=inputs.model_dump() if inputs else {},
1335
- files=files,
1336
- get_cache=chat_service.get_cache,
1337
- set_cache=chat_service.set_cache,
1338
- event_manager=event_manager,
1339
- )
1340
-
1341
- next_runnable_vertices = await self.get_next_runnable_vertices(
1342
- self._lock, vertex=vertex_build_result.vertex, cache=False
1343
- )
1344
- if self.stop_vertex and self.stop_vertex in next_runnable_vertices:
1345
- next_runnable_vertices = [self.stop_vertex]
1346
- self.extend_run_queue(next_runnable_vertices)
1347
- self.reset_inactivated_vertices()
1348
- self.reset_activated_vertices()
1349
-
1350
- await chat_service.set_cache(str(self.flow_id or self._run_id), self)
1351
- self._record_snapshot(vertex_id)
1352
- return vertex_build_result
1353
-
1354
- def get_snapshot(self):
1355
- return copy.deepcopy(
1356
- {
1357
- "run_manager": self.run_manager.to_dict(),
1358
- "run_queue": self._run_queue,
1359
- "vertices_layers": self.vertices_layers,
1360
- "first_layer": self.first_layer,
1361
- "inactive_vertices": self.inactive_vertices,
1362
- "activated_vertices": self.activated_vertices,
1363
- }
1364
- )
1365
-
1366
- def _record_snapshot(self, vertex_id: str | None = None) -> None:
1367
- self._snapshots.append(self.get_snapshot())
1368
- if vertex_id:
1369
- self._call_order.append(vertex_id)
1370
-
1371
- def step(
1372
- self,
1373
- inputs: InputValueRequest | None = None,
1374
- files: list[str] | None = None,
1375
- user_id: str | None = None,
1376
- ):
1377
- """Runs the next vertex in the graph.
1378
-
1379
- Note:
1380
- This function is a synchronous wrapper around `astep`.
1381
- It creates an event loop if one does not exist.
1382
-
1383
- Args:
1384
- inputs: The inputs for the vertex. Defaults to None.
1385
- files: The files for the vertex. Defaults to None.
1386
- user_id: The user ID. Defaults to None.
1387
- """
1388
- return run_until_complete(self.astep(inputs, files, user_id))
1389
-
1390
- async def build_vertex(
1391
- self,
1392
- vertex_id: str,
1393
- *,
1394
- get_cache: GetCache | None = None,
1395
- set_cache: SetCache | None = None,
1396
- inputs_dict: dict[str, str] | None = None,
1397
- files: list[str] | None = None,
1398
- user_id: str | None = None,
1399
- fallback_to_env_vars: bool = False,
1400
- event_manager: EventManager | None = None,
1401
- ) -> VertexBuildResult:
1402
- """Builds a vertex in the graph.
1403
-
1404
- Args:
1405
- vertex_id (str): The ID of the vertex to build.
1406
- get_cache (GetCache): A coroutine to get the cache.
1407
- set_cache (SetCache): A coroutine to set the cache.
1408
- inputs_dict (Optional[Dict[str, str]]): Optional dictionary of inputs for the vertex. Defaults to None.
1409
- files: (Optional[List[str]]): Optional list of files. Defaults to None.
1410
- user_id (Optional[str]): Optional user ID. Defaults to None.
1411
- fallback_to_env_vars (bool): Whether to fallback to environment variables. Defaults to False.
1412
- event_manager (Optional[EventManager]): Optional event manager. Defaults to None.
1413
-
1414
- Returns:
1415
- Tuple: A tuple containing the next runnable vertices, top level vertices, result dictionary,
1416
- parameters, validity flag, artifacts, and the built vertex.
1417
-
1418
- Raises:
1419
- ValueError: If no result is found for the vertex.
1420
- """
1421
- vertex = self.get_vertex(vertex_id)
1422
- self.run_manager.add_to_vertices_being_run(vertex_id)
1423
- try:
1424
- params = ""
1425
- should_build = False
1426
- if not vertex.frozen:
1427
- should_build = True
1428
- else:
1429
- # Check the cache for the vertex
1430
- if get_cache is not None:
1431
- cached_result = await get_cache(key=vertex.id)
1432
- else:
1433
- cached_result = CacheMiss()
1434
- if isinstance(cached_result, CacheMiss):
1435
- should_build = True
1436
- else:
1437
- try:
1438
- cached_vertex_dict = cached_result["result"]
1439
- # Now set update the vertex with the cached vertex
1440
- vertex.built = cached_vertex_dict["built"]
1441
- vertex.artifacts = cached_vertex_dict["artifacts"]
1442
- vertex.built_object = cached_vertex_dict["built_object"]
1443
- vertex.built_result = cached_vertex_dict["built_result"]
1444
- vertex.full_data = cached_vertex_dict["full_data"]
1445
- vertex.results = cached_vertex_dict["results"]
1446
- try:
1447
- vertex.finalize_build()
1448
-
1449
- if vertex.result is not None:
1450
- vertex.result.used_frozen_result = True
1451
- except Exception: # noqa: BLE001
1452
- logger.debug("Error finalizing build", exc_info=True)
1453
- should_build = True
1454
- except KeyError:
1455
- should_build = True
1456
-
1457
- if should_build:
1458
- await vertex.build(
1459
- user_id=user_id,
1460
- inputs=inputs_dict,
1461
- fallback_to_env_vars=fallback_to_env_vars,
1462
- files=files,
1463
- event_manager=event_manager,
1464
- )
1465
- if set_cache is not None:
1466
- vertex_dict = {
1467
- "built": vertex.built,
1468
- "results": vertex.results,
1469
- "artifacts": vertex.artifacts,
1470
- "built_object": vertex.built_object,
1471
- "built_result": vertex.built_result,
1472
- "full_data": vertex.full_data,
1473
- }
1474
-
1475
- await set_cache(key=vertex.id, data=vertex_dict)
1476
-
1477
- except Exception as exc:
1478
- if not isinstance(exc, ComponentBuildError):
1479
- await logger.aexception("Error building Component")
1480
- raise
1481
-
1482
- if vertex.result is not None:
1483
- params = f"{vertex.built_object_repr()}{params}"
1484
- valid = True
1485
- result_dict = vertex.result
1486
- artifacts = vertex.artifacts
1487
- else:
1488
- msg = f"Error building Component: no result found for vertex {vertex_id}"
1489
- raise ValueError(msg)
1490
-
1491
- return VertexBuildResult(
1492
- result_dict=result_dict, params=params, valid=valid, artifacts=artifacts, vertex=vertex
1493
- )
1494
-
1495
- def get_vertex_edges(
1496
- self,
1497
- vertex_id: str,
1498
- *,
1499
- is_target: bool | None = None,
1500
- is_source: bool | None = None,
1501
- ) -> list[CycleEdge]:
1502
- """Returns a list of edges for a given vertex."""
1503
- # The idea here is to return the edges that have the vertex_id as source or target
1504
- # or both
1505
- return [
1506
- edge
1507
- for edge in self.edges
1508
- if (edge.source_id == vertex_id and is_source is not False)
1509
- or (edge.target_id == vertex_id and is_target is not False)
1510
- ]
1511
-
1512
- def get_vertices_with_target(self, vertex_id: str) -> list[Vertex]:
1513
- """Returns the vertices connected to a vertex."""
1514
- vertices: list[Vertex] = []
1515
- for edge in self.edges:
1516
- if edge.target_id == vertex_id:
1517
- vertex = self.get_vertex(edge.source_id)
1518
- if vertex is None:
1519
- continue
1520
- vertices.append(vertex)
1521
- return vertices
1522
-
1523
- async def process(
1524
- self,
1525
- *,
1526
- fallback_to_env_vars: bool,
1527
- start_component_id: str | None = None,
1528
- event_manager: EventManager | None = None,
1529
- ) -> Graph:
1530
- """Processes the graph with vertices in each layer run in parallel."""
1531
- has_webhook_component = "webhook" in start_component_id.lower() if start_component_id else False
1532
- first_layer = self.sort_vertices(start_component_id=start_component_id)
1533
- vertex_task_run_count: dict[str, int] = {}
1534
- to_process = deque(first_layer)
1535
- layer_index = 0
1536
- chat_service = get_chat_service()
1537
- await self.initialize_run()
1538
- lock = asyncio.Lock()
1539
- while to_process:
1540
- current_batch = list(to_process) # Copy current deque items to a list
1541
- to_process.clear() # Clear the deque for new items
1542
- tasks = []
1543
- for vertex_id in current_batch:
1544
- vertex = self.get_vertex(vertex_id)
1545
- task = asyncio.create_task(
1546
- self.build_vertex(
1547
- vertex_id=vertex_id,
1548
- user_id=self.user_id,
1549
- inputs_dict={},
1550
- fallback_to_env_vars=fallback_to_env_vars,
1551
- get_cache=chat_service.get_cache,
1552
- set_cache=chat_service.set_cache,
1553
- event_manager=event_manager,
1554
- ),
1555
- name=f"{vertex.id} Run {vertex_task_run_count.get(vertex_id, 0)}",
1556
- )
1557
- tasks.append(task)
1558
- vertex_task_run_count[vertex_id] = vertex_task_run_count.get(vertex_id, 0) + 1
1559
-
1560
- await logger.adebug(f"Running layer {layer_index} with {len(tasks)} tasks, {current_batch}")
1561
- try:
1562
- next_runnable_vertices = await self._execute_tasks(
1563
- tasks, lock=lock, has_webhook_component=has_webhook_component
1564
- )
1565
- except Exception:
1566
- await logger.aexception(f"Error executing tasks in layer {layer_index}")
1567
- raise
1568
- if not next_runnable_vertices:
1569
- break
1570
- to_process.extend(next_runnable_vertices)
1571
- layer_index += 1
1572
-
1573
- await logger.adebug("Graph processing complete")
1574
- return self
1575
-
1576
- def find_next_runnable_vertices(self, vertex_successors_ids: list[str]) -> list[str]:
1577
- """Determines the next set of runnable vertices from a list of successor vertex IDs.
1578
-
1579
- For each successor, if it is not runnable, recursively finds its runnable
1580
- predecessors; otherwise, includes the successor itself. Returns a sorted list of all such vertex IDs.
1581
- """
1582
- next_runnable_vertices = set()
1583
- for v_id in sorted(vertex_successors_ids):
1584
- if not self.is_vertex_runnable(v_id):
1585
- next_runnable_vertices.update(self.find_runnable_predecessors_for_successor(v_id))
1586
- else:
1587
- next_runnable_vertices.add(v_id)
1588
-
1589
- return sorted(next_runnable_vertices)
1590
-
1591
- async def get_next_runnable_vertices(self, lock: asyncio.Lock, vertex: Vertex, *, cache: bool = True) -> list[str]:
1592
- """Determines the next set of runnable vertex IDs after a vertex completes execution.
1593
-
1594
- If the completed vertex is a state vertex, any recently activated state vertices are also included.
1595
- Updates the run manager to reflect the new runnable state and optionally caches the updated graph state.
1596
-
1597
- Args:
1598
- lock: An asyncio lock for thread-safe updates.
1599
- vertex: The vertex that has just finished execution.
1600
- cache: If True, caches the updated graph state.
1601
-
1602
- Returns:
1603
- A list of vertex IDs that are ready to be executed next.
1604
- """
1605
- v_id = vertex.id
1606
- v_successors_ids = vertex.successors_ids
1607
- self.run_manager.ran_at_least_once.add(v_id)
1608
- async with lock:
1609
- self.run_manager.remove_vertex_from_runnables(v_id)
1610
- next_runnable_vertices = self.find_next_runnable_vertices(v_successors_ids)
1611
-
1612
- for next_v_id in set(next_runnable_vertices): # Use set to avoid duplicates
1613
- if next_v_id == v_id:
1614
- next_runnable_vertices.remove(v_id)
1615
- else:
1616
- self.run_manager.add_to_vertices_being_run(next_v_id)
1617
- if cache and self.flow_id is not None:
1618
- set_cache_coro = partial(get_chat_service().set_cache, key=self.flow_id)
1619
- await set_cache_coro(data=self, lock=lock)
1620
- if vertex.is_state:
1621
- next_runnable_vertices.extend(self.activated_vertices)
1622
- return next_runnable_vertices
1623
-
1624
- async def _log_vertex_build_from_exception(self, vertex_id: str, result: Exception) -> None:
1625
- """Logs detailed information about a vertex build exception.
1626
-
1627
- Formats the exception message and stack trace, constructs an error output,
1628
- and records the failure using the vertex build logging system.
1629
- """
1630
- if isinstance(result, ComponentBuildError):
1631
- params = result.message
1632
- tb = result.formatted_traceback
1633
- else:
1634
- from langflow.api.utils import format_exception_message
1635
-
1636
- tb = traceback.format_exc()
1637
- await logger.aexception("Error building Component")
1638
-
1639
- params = format_exception_message(result)
1640
- message = {"errorMessage": params, "stackTrace": tb}
1641
- vertex = self.get_vertex(vertex_id)
1642
- output_label = vertex.outputs[0]["name"] if vertex.outputs else "output"
1643
- outputs = {output_label: OutputValue(message=message, type="error")}
1644
- result_data_response = {
1645
- "results": {},
1646
- "outputs": outputs,
1647
- "logs": {},
1648
- "message": {},
1649
- "artifacts": {},
1650
- "timedelta": None,
1651
- "duration": None,
1652
- "used_frozen_result": False,
1653
- }
1654
-
1655
- await log_vertex_build(
1656
- flow_id=self.flow_id or "",
1657
- vertex_id=vertex_id or "errors",
1658
- valid=False,
1659
- params=params,
1660
- data=result_data_response,
1661
- artifacts={},
1662
- )
1663
-
1664
- async def _execute_tasks(
1665
- self, tasks: list[asyncio.Task], lock: asyncio.Lock, *, has_webhook_component: bool = False
1666
- ) -> list[str]:
1667
- """Executes tasks in parallel, handling exceptions for each task.
1668
-
1669
- Args:
1670
- tasks: List of tasks to execute
1671
- lock: Async lock for synchronization
1672
- has_webhook_component: Whether the graph has a webhook component
1673
- """
1674
- results = []
1675
- completed_tasks = await asyncio.gather(*tasks, return_exceptions=True)
1676
- vertices: list[Vertex] = []
1677
-
1678
- for i, result in enumerate(completed_tasks):
1679
- task_name = tasks[i].get_name()
1680
- vertex_id = tasks[i].get_name().split(" ")[0]
1681
-
1682
- if isinstance(result, Exception):
1683
- await logger.aerror(f"Task {task_name} failed with exception: {result}")
1684
- if has_webhook_component:
1685
- await self._log_vertex_build_from_exception(vertex_id, result)
1686
-
1687
- # Cancel all remaining tasks
1688
- for t in tasks[i + 1 :]:
1689
- t.cancel()
1690
- raise result
1691
- if isinstance(result, VertexBuildResult):
1692
- if self.flow_id is not None:
1693
- await log_vertex_build(
1694
- flow_id=self.flow_id,
1695
- vertex_id=result.vertex.id,
1696
- valid=result.valid,
1697
- params=result.params,
1698
- data=result.result_dict,
1699
- artifacts=result.artifacts,
1700
- )
1701
-
1702
- vertices.append(result.vertex)
1703
- else:
1704
- msg = f"Invalid result from task {task_name}: {result}"
1705
- raise TypeError(msg)
1706
-
1707
- for v in vertices:
1708
- # set all executed vertices as non-runnable to not run them again.
1709
- # they could be calculated as predecessor or successors of parallel vertices
1710
- # This could usually happen with input vertices like ChatInput
1711
- self.run_manager.remove_vertex_from_runnables(v.id)
1712
-
1713
- await logger.adebug(f"Vertex {v.id}, result: {v.built_result}, object: {v.built_object}")
1714
-
1715
- for v in vertices:
1716
- next_runnable_vertices = await self.get_next_runnable_vertices(lock, vertex=v, cache=False)
1717
- results.extend(next_runnable_vertices)
1718
- return list(set(results))
1719
-
1720
- def topological_sort(self) -> list[Vertex]:
1721
- """Performs a topological sort of the vertices in the graph.
1722
-
1723
- Returns:
1724
- List[Vertex]: A list of vertices in topological order.
1725
-
1726
- Raises:
1727
- ValueError: If the graph contains a cycle.
1728
- """
1729
- # States: 0 = unvisited, 1 = visiting, 2 = visited
1730
- state = dict.fromkeys(self.vertices, 0)
1731
- sorted_vertices = []
1732
-
1733
- def dfs(vertex) -> None:
1734
- if state[vertex] == 1:
1735
- # We have a cycle
1736
- msg = "Graph contains a cycle, cannot perform topological sort"
1737
- raise ValueError(msg)
1738
- if state[vertex] == 0:
1739
- state[vertex] = 1
1740
- for edge in vertex.edges:
1741
- if edge.source_id == vertex.id:
1742
- dfs(self.get_vertex(edge.target_id))
1743
- state[vertex] = 2
1744
- sorted_vertices.append(vertex)
1745
-
1746
- # Visit each vertex
1747
- for vertex in self.vertices:
1748
- if state[vertex] == 0:
1749
- dfs(vertex)
1750
-
1751
- return list(reversed(sorted_vertices))
1752
-
1753
- def generator_build(self) -> Generator[Vertex, None, None]:
1754
- """Builds each vertex in the graph and yields it."""
1755
- sorted_vertices = self.topological_sort()
1756
- logger.debug("There are %s vertices in the graph", len(sorted_vertices))
1757
- yield from sorted_vertices
1758
-
1759
- def get_predecessors(self, vertex):
1760
- """Returns the predecessors of a vertex."""
1761
- return [self.get_vertex(source_id) for source_id in self.predecessor_map.get(vertex.id, [])]
1762
-
1763
- def get_all_successors(self, vertex: Vertex, *, recursive=True, flat=True, visited=None):
1764
- """Returns all successors of a given vertex, optionally recursively and as a flat or nested list.
1765
-
1766
- Args:
1767
- vertex: The vertex whose successors are to be retrieved.
1768
- recursive: If True, retrieves successors recursively; otherwise, only immediate successors.
1769
- flat: If True, returns a flat list of successors; if False, returns a nested list structure.
1770
- visited: Internal set used to track visited vertices and prevent cycles.
1771
-
1772
- Returns:
1773
- A list of successor vertices, either flat or nested depending on the `flat` parameter.
1774
- """
1775
- if visited is None:
1776
- visited = set()
1777
-
1778
- # Prevent revisiting vertices to avoid infinite loops in cyclic graphs
1779
- if vertex in visited:
1780
- return []
1781
-
1782
- visited.add(vertex)
1783
-
1784
- successors = vertex.successors
1785
- if not successors:
1786
- return []
1787
-
1788
- successors_result = []
1789
-
1790
- for successor in successors:
1791
- if recursive:
1792
- next_successors = self.get_all_successors(successor, recursive=recursive, flat=flat, visited=visited)
1793
- if flat:
1794
- successors_result.extend(next_successors)
1795
- else:
1796
- successors_result.append(next_successors)
1797
- if flat:
1798
- successors_result.append(successor)
1799
- else:
1800
- successors_result.append([successor])
1801
-
1802
- if not flat and successors_result:
1803
- return [successors, *successors_result]
1804
-
1805
- return successors_result
1806
-
1807
- def get_successors(self, vertex: Vertex) -> list[Vertex]:
1808
- """Returns the immediate successor vertices of the given vertex.
1809
-
1810
- Args:
1811
- vertex: The vertex whose successors are to be retrieved.
1812
-
1813
- Returns:
1814
- A list of vertices that are direct successors of the specified vertex.
1815
- """
1816
- return [self.get_vertex(target_id) for target_id in self.successor_map.get(vertex.id, set())]
1817
-
1818
- def get_all_predecessors(self, vertex: Vertex, *, recursive: bool = True) -> list[Vertex]:
1819
- """Retrieves all predecessor vertices of a given vertex.
1820
-
1821
- If `recursive` is True, returns both direct and indirect predecessors by
1822
- traversing the graph recursively. If False, returns only the immediate predecessors.
1823
- """
1824
- _predecessors = self.predecessor_map.get(vertex.id, [])
1825
- predecessors = [self.get_vertex(v_id) for v_id in _predecessors]
1826
- if recursive:
1827
- for predecessor in _predecessors:
1828
- predecessors.extend(self.get_all_predecessors(self.get_vertex(predecessor), recursive=recursive))
1829
- else:
1830
- predecessors.extend([self.get_vertex(predecessor) for predecessor in _predecessors])
1831
- return predecessors
1832
-
1833
- def get_vertex_neighbors(self, vertex: Vertex) -> dict[Vertex, int]:
1834
- """Returns a dictionary mapping each direct neighbor of a vertex to the count of connecting edges.
1835
-
1836
- A neighbor is any vertex directly connected to the input vertex, either as a source or target.
1837
- The count reflects the number of edges between the input vertex and each neighbor.
1838
- """
1839
- neighbors: dict[Vertex, int] = {}
1840
- for edge in self.edges:
1841
- if edge.source_id == vertex.id:
1842
- neighbor = self.get_vertex(edge.target_id)
1843
- if neighbor is None:
1844
- continue
1845
- if neighbor not in neighbors:
1846
- neighbors[neighbor] = 0
1847
- neighbors[neighbor] += 1
1848
- elif edge.target_id == vertex.id:
1849
- neighbor = self.get_vertex(edge.source_id)
1850
- if neighbor is None:
1851
- continue
1852
- if neighbor not in neighbors:
1853
- neighbors[neighbor] = 0
1854
- neighbors[neighbor] += 1
1855
- return neighbors
1856
-
1857
- @property
1858
- def cycles(self):
1859
- if self._cycles is None:
1860
- if self._start is None:
1861
- self._cycles = []
1862
- else:
1863
- entry_vertex = self._start._id
1864
- edges = [(e["data"]["sourceHandle"]["id"], e["data"]["targetHandle"]["id"]) for e in self._edges]
1865
- self._cycles = find_all_cycle_edges(entry_vertex, edges)
1866
- return self._cycles
1867
-
1868
- @property
1869
- def cycle_vertices(self):
1870
- if self._cycle_vertices is None:
1871
- edges = self._get_edges_as_list_of_tuples()
1872
- self._cycle_vertices = set(find_cycle_vertices(edges))
1873
- return self._cycle_vertices
1874
-
1875
- def _build_edges(self) -> list[CycleEdge]:
1876
- """Builds the edges of the graph."""
1877
- # Edge takes two vertices as arguments, so we need to build the vertices first
1878
- # and then build the edges
1879
- # if we can't find a vertex, we raise an error
1880
- edges: set[CycleEdge | Edge] = set()
1881
- for edge in self._edges:
1882
- new_edge = self.build_edge(edge)
1883
- edges.add(new_edge)
1884
- if self.vertices and not edges:
1885
- logger.warning("Graph has vertices but no edges")
1886
- return list(cast("Iterable[CycleEdge]", edges))
1887
-
1888
- def build_edge(self, edge: EdgeData) -> CycleEdge | Edge:
1889
- source = self.get_vertex(edge["source"])
1890
- target = self.get_vertex(edge["target"])
1891
-
1892
- if source is None:
1893
- msg = f"Source vertex {edge['source']} not found"
1894
- raise ValueError(msg)
1895
- if target is None:
1896
- msg = f"Target vertex {edge['target']} not found"
1897
- raise ValueError(msg)
1898
- if any(v in self.cycle_vertices for v in [source.id, target.id]):
1899
- new_edge: CycleEdge | Edge = CycleEdge(source, target, edge)
1900
- else:
1901
- new_edge = Edge(source, target, edge)
1902
- return new_edge
1903
-
1904
- @staticmethod
1905
- def _get_vertex_class(node_type: str, node_base_type: str, node_id: str) -> type[Vertex]:
1906
- """Returns the node class based on the node type."""
1907
- # First we check for the node_base_type
1908
- node_name = node_id.split("-")[0]
1909
- if node_name in InterfaceComponentTypes:
1910
- return InterfaceVertex
1911
- if node_name in {"SharedState", "Notify", "Listen"}:
1912
- return StateVertex
1913
- if node_base_type in lazy_load_vertex_dict.vertex_type_map:
1914
- return lazy_load_vertex_dict.vertex_type_map[node_base_type]
1915
- if node_name in lazy_load_vertex_dict.vertex_type_map:
1916
- return lazy_load_vertex_dict.vertex_type_map[node_name]
1917
-
1918
- if node_type in lazy_load_vertex_dict.vertex_type_map:
1919
- return lazy_load_vertex_dict.vertex_type_map[node_type]
1920
- return Vertex
1921
-
1922
- def _build_vertices(self) -> list[Vertex]:
1923
- """Builds the vertices of the graph."""
1924
- vertices: list[Vertex] = []
1925
- for frontend_data in self._vertices:
1926
- if frontend_data.get("type") == NodeTypeEnum.NoteNode:
1927
- continue
1928
- try:
1929
- vertex_instance = self.get_vertex(frontend_data["id"])
1930
- except ValueError:
1931
- vertex_instance = self._create_vertex(frontend_data)
1932
- vertices.append(vertex_instance)
1933
-
1934
- return vertices
1935
-
1936
- def _create_vertex(self, frontend_data: NodeData):
1937
- vertex_data = frontend_data["data"]
1938
- vertex_type: str = vertex_data["type"]
1939
- vertex_base_type: str = vertex_data["node"]["template"]["_type"]
1940
- if "id" not in vertex_data:
1941
- msg = f"Vertex data for {vertex_data['display_name']} does not contain an id"
1942
- raise ValueError(msg)
1943
-
1944
- vertex_class = self._get_vertex_class(vertex_type, vertex_base_type, vertex_data["id"])
1945
-
1946
- vertex_instance = vertex_class(frontend_data, graph=self)
1947
- vertex_instance.set_top_level(self.top_level_vertices)
1948
- return vertex_instance
1949
-
1950
- def prepare(self, stop_component_id: str | None = None, start_component_id: str | None = None):
1951
- self.initialize()
1952
- if stop_component_id and start_component_id:
1953
- msg = "You can only provide one of stop_component_id or start_component_id"
1954
- raise ValueError(msg)
1955
-
1956
- if stop_component_id or start_component_id:
1957
- try:
1958
- first_layer = self.sort_vertices(stop_component_id, start_component_id)
1959
- except Exception: # noqa: BLE001
1960
- logger.exception("Error sorting vertices")
1961
- first_layer = self.sort_vertices()
1962
- else:
1963
- first_layer = self.sort_vertices()
1964
-
1965
- for vertex_id in first_layer:
1966
- self.run_manager.add_to_vertices_being_run(vertex_id)
1967
- if vertex_id in self.cycle_vertices:
1968
- self.run_manager.add_to_cycle_vertices(vertex_id)
1969
- self._first_layer = sorted(first_layer)
1970
- self._run_queue = deque(self._first_layer)
1971
- self._prepared = True
1972
- self._record_snapshot()
1973
- return self
1974
-
1975
- @staticmethod
1976
- def get_children_by_vertex_type(vertex: Vertex, vertex_type: str) -> list[Vertex]:
1977
- """Returns the children of a vertex based on the vertex type."""
1978
- children = []
1979
- vertex_types = [vertex.data["type"]]
1980
- if "node" in vertex.data:
1981
- vertex_types += vertex.data["node"]["base_classes"]
1982
- if vertex_type in vertex_types:
1983
- children.append(vertex)
1984
- return children
1985
-
1986
- def __repr__(self) -> str:
1987
- vertex_ids = [vertex.id for vertex in self.vertices]
1988
- edges_repr = "\n".join([f" {edge.source_id} --> {edge.target_id}" for edge in self.edges])
1989
-
1990
- return (
1991
- f"Graph Representation:\n"
1992
- f"----------------------\n"
1993
- f"Vertices ({len(vertex_ids)}):\n"
1994
- f" {', '.join(map(str, vertex_ids))}\n\n"
1995
- f"Edges ({len(self.edges)}):\n"
1996
- f"{edges_repr}"
1997
- )
1998
-
1999
- def __hash__(self) -> int:
2000
- """Return hash of the graph based on its string representation."""
2001
- return hash(self.__repr__())
2002
-
2003
- def get_vertex_predecessors_ids(self, vertex_id: str) -> list[str]:
2004
- """Get the predecessor IDs of a vertex."""
2005
- return [v.id for v in self.get_predecessors(self.get_vertex(vertex_id))]
2006
-
2007
- def get_vertex_successors_ids(self, vertex_id: str) -> list[str]:
2008
- """Get the successor IDs of a vertex."""
2009
- return [v.id for v in self.get_vertex(vertex_id).successors]
2010
-
2011
- def get_vertex_input_status(self, vertex_id: str) -> bool:
2012
- """Check if a vertex is an input vertex."""
2013
- return self.get_vertex(vertex_id).is_input
2014
-
2015
- def get_parent_map(self) -> dict[str, str | None]:
2016
- """Get the parent node map for all vertices."""
2017
- return {vertex.id: vertex.parent_node_id for vertex in self.vertices}
2018
-
2019
- def get_vertex_ids(self) -> list[str]:
2020
- """Get all vertex IDs in the graph."""
2021
- return [vertex.id for vertex in self.vertices]
2022
-
2023
- def sort_vertices(
2024
- self,
2025
- stop_component_id: str | None = None,
2026
- start_component_id: str | None = None,
2027
- ) -> list[str]:
2028
- """Sorts the vertices in the graph."""
2029
- self.mark_all_vertices("ACTIVE")
2030
-
2031
- first_layer, remaining_layers = get_sorted_vertices(
2032
- vertices_ids=self.get_vertex_ids(),
2033
- cycle_vertices=self.cycle_vertices,
2034
- stop_component_id=stop_component_id,
2035
- start_component_id=start_component_id,
2036
- graph_dict=self.__to_dict(),
2037
- in_degree_map=self.in_degree_map,
2038
- successor_map=self.successor_map,
2039
- predecessor_map=self.predecessor_map,
2040
- is_input_vertex=self.get_vertex_input_status,
2041
- get_vertex_predecessors=self.get_vertex_predecessors_ids,
2042
- get_vertex_successors=self.get_vertex_successors_ids,
2043
- is_cyclic=self.is_cyclic,
2044
- )
2045
-
2046
- self.increment_run_count()
2047
- self._sorted_vertices_layers = [first_layer, *remaining_layers]
2048
- self.vertices_layers = remaining_layers
2049
- self.vertices_to_run = set(chain.from_iterable([first_layer, *remaining_layers]))
2050
- self.build_run_map()
2051
- self._first_layer = first_layer
2052
- return first_layer
2053
-
2054
- @staticmethod
2055
- def sort_interface_components_first(vertices_layers: list[list[str]]) -> list[list[str]]:
2056
- """Sorts the vertices in the graph so that vertices containing ChatInput or ChatOutput come first."""
2057
-
2058
- def contains_interface_component(vertex):
2059
- return any(component.value in vertex for component in InterfaceComponentTypes)
2060
-
2061
- # Sort each inner list so that vertices containing ChatInput or ChatOutput come first
2062
- return [
2063
- sorted(
2064
- inner_list,
2065
- key=lambda vertex: not contains_interface_component(vertex),
2066
- )
2067
- for inner_list in vertices_layers
2068
- ]
2069
-
2070
- def sort_by_avg_build_time(self, vertices_layers: list[list[str]]) -> list[list[str]]:
2071
- """Sorts the vertices in the graph so that vertices with the lowest average build time come first."""
2072
-
2073
- def sort_layer_by_avg_build_time(vertices_ids: list[str]) -> list[str]:
2074
- """Sorts the vertices in the graph so that vertices with the lowest average build time come first."""
2075
- if len(vertices_ids) == 1:
2076
- return vertices_ids
2077
- vertices_ids.sort(key=lambda vertex_id: self.get_vertex(vertex_id).avg_build_time)
2078
-
2079
- return vertices_ids
2080
-
2081
- return [sort_layer_by_avg_build_time(layer) for layer in vertices_layers]
2082
-
2083
- def is_vertex_runnable(self, vertex_id: str) -> bool:
2084
- """Returns whether a vertex is runnable."""
2085
- is_active = self.get_vertex(vertex_id).is_active()
2086
- is_loop = self.get_vertex(vertex_id).is_loop
2087
- return self.run_manager.is_vertex_runnable(vertex_id, is_active=is_active, is_loop=is_loop)
2088
-
2089
- def build_run_map(self) -> None:
2090
- """Builds the run map for the graph.
2091
-
2092
- This method is responsible for building the run map for the graph,
2093
- which maps each node in the graph to its corresponding run function.
2094
- """
2095
- self.run_manager.build_run_map(predecessor_map=self.predecessor_map, vertices_to_run=self.vertices_to_run)
2096
-
2097
- def find_runnable_predecessors_for_successors(self, vertex_id: str) -> list[str]:
2098
- """For each successor of the current vertex, find runnable predecessors if any.
2099
-
2100
- This checks the direct predecessors of each successor to identify any that are
2101
- immediately runnable, expanding the search to ensure progress can be made.
2102
- """
2103
- runnable_vertices = []
2104
- for successor_id in self.run_manager.run_map.get(vertex_id, []):
2105
- runnable_vertices.extend(self.find_runnable_predecessors_for_successor(successor_id))
2106
-
2107
- return sorted(runnable_vertices)
2108
-
2109
- def find_runnable_predecessors_for_successor(self, vertex_id: str) -> list[str]:
2110
- runnable_vertices = []
2111
- visited = set()
2112
-
2113
- def find_runnable_predecessors(predecessor_id: str) -> None:
2114
- if predecessor_id in visited:
2115
- return
2116
- visited.add(predecessor_id)
2117
- predecessor_vertex = self.get_vertex(predecessor_id)
2118
- is_active = predecessor_vertex.is_active()
2119
- is_loop = predecessor_vertex.is_loop
2120
- if self.run_manager.is_vertex_runnable(predecessor_id, is_active=is_active, is_loop=is_loop):
2121
- runnable_vertices.append(predecessor_id)
2122
- else:
2123
- for pred_pred_id in self.run_manager.run_predecessors.get(predecessor_id, []):
2124
- find_runnable_predecessors(pred_pred_id)
2125
-
2126
- for predecessor_id in self.run_manager.run_predecessors.get(vertex_id, []):
2127
- find_runnable_predecessors(predecessor_id)
2128
- return runnable_vertices
2129
-
2130
- def remove_from_predecessors(self, vertex_id: str) -> None:
2131
- self.run_manager.remove_from_predecessors(vertex_id)
2132
-
2133
- def remove_vertex_from_runnables(self, vertex_id: str) -> None:
2134
- self.run_manager.remove_vertex_from_runnables(vertex_id)
2135
-
2136
- def get_top_level_vertices(self, vertices_ids):
2137
- """Retrieves the top-level vertices from the given graph based on the provided vertex IDs.
2138
-
2139
- Args:
2140
- vertices_ids (list): A list of vertex IDs.
2141
-
2142
- Returns:
2143
- list: A list of top-level vertex IDs.
2144
-
2145
- """
2146
- top_level_vertices = []
2147
- for vertex_id in vertices_ids:
2148
- vertex = self.get_vertex(vertex_id)
2149
- if vertex.parent_is_top_level:
2150
- top_level_vertices.append(vertex.parent_node_id)
2151
- else:
2152
- top_level_vertices.append(vertex_id)
2153
- return top_level_vertices
2154
-
2155
- def build_in_degree(self, edges: list[CycleEdge]) -> dict[str, int]:
2156
- in_degree: dict[str, int] = defaultdict(int)
2157
-
2158
- for edge in edges:
2159
- # We don't need to count if a Component connects more than one
2160
- # time to the same vertex.
2161
- in_degree[edge.target_id] += 1
2162
- for vertex in self.vertices:
2163
- if vertex.id not in in_degree:
2164
- in_degree[vertex.id] = 0
2165
- return in_degree
2166
-
2167
- @staticmethod
2168
- def build_adjacency_maps(edges: list[CycleEdge]) -> tuple[dict[str, list[str]], dict[str, list[str]]]:
2169
- """Returns the adjacency maps for the graph."""
2170
- predecessor_map: dict[str, list[str]] = defaultdict(list)
2171
- successor_map: dict[str, list[str]] = defaultdict(list)
2172
- for edge in edges:
2173
- predecessor_map[edge.target_id].append(edge.source_id)
2174
- successor_map[edge.source_id].append(edge.target_id)
2175
- return predecessor_map, successor_map
2176
-
2177
- def __to_dict(self) -> dict[str, dict[str, list[str]]]:
2178
- """Converts the graph to a dictionary."""
2179
- result: dict = {}
2180
- for vertex in self.vertices:
2181
- vertex_id = vertex.id
2182
- sucessors = [i.id for i in self.get_all_successors(vertex)]
2183
- predecessors = [i.id for i in self.get_predecessors(vertex)]
2184
- result |= {vertex_id: {"successors": sucessors, "predecessors": predecessors}}
2185
- return result