flowent 0.0.1 → 0.0.5

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 (484) hide show
  1. package/README.md +20 -9
  2. package/backend/.python-version +1 -0
  3. package/backend/README.md +74 -0
  4. package/backend/pyproject.toml +58 -0
  5. package/backend/src/flowent/__init__.py +3 -0
  6. package/backend/src/flowent/__pycache__/__init__.cpython-313.pyc +0 -0
  7. package/backend/src/flowent/__pycache__/_version.cpython-313.pyc +0 -0
  8. package/backend/src/flowent/__pycache__/access.cpython-313.pyc +0 -0
  9. package/backend/src/flowent/__pycache__/agent.cpython-313.pyc +0 -0
  10. package/backend/src/flowent/__pycache__/assistant_commands.cpython-313.pyc +0 -0
  11. package/backend/src/flowent/__pycache__/cli.cpython-313.pyc +0 -0
  12. package/backend/src/flowent/__pycache__/config.cpython-313.pyc +0 -0
  13. package/backend/src/flowent/__pycache__/events.cpython-313.pyc +0 -0
  14. package/backend/src/flowent/__pycache__/graph_runtime.cpython-313.pyc +0 -0
  15. package/backend/src/flowent/__pycache__/graph_service.cpython-313.pyc +0 -0
  16. package/backend/src/flowent/__pycache__/image_assets.cpython-313.pyc +0 -0
  17. package/backend/src/flowent/__pycache__/logging.cpython-313.pyc +0 -0
  18. package/backend/src/flowent/__pycache__/main.cpython-313.pyc +0 -0
  19. package/backend/src/flowent/__pycache__/mcp_service.cpython-313.pyc +0 -0
  20. package/backend/src/flowent/__pycache__/model_metadata.cpython-313.pyc +0 -0
  21. package/backend/src/flowent/__pycache__/network.cpython-313.pyc +0 -0
  22. package/backend/src/flowent/__pycache__/observability_service.cpython-313.pyc +0 -0
  23. package/backend/src/flowent/__pycache__/registry.cpython-313.pyc +0 -0
  24. package/backend/src/flowent/__pycache__/role_management.cpython-313.pyc +0 -0
  25. package/backend/src/flowent/__pycache__/runtime.cpython-313.pyc +0 -0
  26. package/backend/src/flowent/__pycache__/sandbox.cpython-313.pyc +0 -0
  27. package/backend/src/flowent/__pycache__/security.cpython-313.pyc +0 -0
  28. package/backend/src/flowent/__pycache__/settings.cpython-313.pyc +0 -0
  29. package/backend/src/flowent/__pycache__/settings_management.cpython-313.pyc +0 -0
  30. package/backend/src/flowent/__pycache__/state_db.cpython-313.pyc +0 -0
  31. package/backend/src/flowent/__pycache__/workspace_store.cpython-313.pyc +0 -0
  32. package/backend/src/flowent/_version.py +7 -0
  33. package/backend/src/flowent/access.py +247 -0
  34. package/backend/src/flowent/agent.py +3120 -0
  35. package/backend/src/flowent/assistant_commands.py +115 -0
  36. package/backend/src/flowent/channels/__init__.py +3 -0
  37. package/backend/src/flowent/channels/__pycache__/__init__.cpython-313.pyc +0 -0
  38. package/backend/src/flowent/channels/__pycache__/telegram.cpython-313.pyc +0 -0
  39. package/backend/src/flowent/channels/telegram.py +615 -0
  40. package/backend/src/flowent/cli.py +85 -0
  41. package/backend/src/flowent/config.py +14 -0
  42. package/backend/src/flowent/dev.py +3 -0
  43. package/backend/src/flowent/events.py +157 -0
  44. package/backend/src/flowent/graph_runtime.py +60 -0
  45. package/backend/src/flowent/graph_service.py +2508 -0
  46. package/backend/src/flowent/image_assets.py +356 -0
  47. package/backend/src/flowent/logging.py +155 -0
  48. package/backend/src/flowent/main.py +124 -0
  49. package/backend/src/flowent/mcp_service.py +1918 -0
  50. package/backend/src/flowent/model_metadata.py +102 -0
  51. package/backend/src/flowent/models/__init__.py +125 -0
  52. package/backend/src/flowent/models/__pycache__/__init__.cpython-313.pyc +0 -0
  53. package/backend/src/flowent/models/__pycache__/agent.cpython-313.pyc +0 -0
  54. package/backend/src/flowent/models/__pycache__/base.cpython-313.pyc +0 -0
  55. package/backend/src/flowent/models/__pycache__/blueprint.cpython-313.pyc +0 -0
  56. package/backend/src/flowent/models/__pycache__/content.cpython-313.pyc +0 -0
  57. package/backend/src/flowent/models/__pycache__/delta.cpython-313.pyc +0 -0
  58. package/backend/src/flowent/models/__pycache__/event.cpython-313.pyc +0 -0
  59. package/backend/src/flowent/models/__pycache__/graph.cpython-313.pyc +0 -0
  60. package/backend/src/flowent/models/__pycache__/history.cpython-313.pyc +0 -0
  61. package/backend/src/flowent/models/__pycache__/llm.cpython-313.pyc +0 -0
  62. package/backend/src/flowent/models/__pycache__/message.cpython-313.pyc +0 -0
  63. package/backend/src/flowent/models/__pycache__/tab.cpython-313.pyc +0 -0
  64. package/backend/src/flowent/models/__pycache__/todo.cpython-313.pyc +0 -0
  65. package/backend/src/flowent/models/agent.py +34 -0
  66. package/backend/src/flowent/models/base.py +24 -0
  67. package/backend/src/flowent/models/blueprint.py +176 -0
  68. package/backend/src/flowent/models/content.py +164 -0
  69. package/backend/src/flowent/models/delta.py +44 -0
  70. package/backend/src/flowent/models/event.py +51 -0
  71. package/backend/src/flowent/models/graph.py +472 -0
  72. package/backend/src/flowent/models/history.py +272 -0
  73. package/backend/src/flowent/models/llm.py +62 -0
  74. package/backend/src/flowent/models/message.py +33 -0
  75. package/backend/src/flowent/models/tab.py +85 -0
  76. package/backend/src/flowent/models/todo.py +10 -0
  77. package/backend/src/flowent/network.py +146 -0
  78. package/backend/src/flowent/observability_service.py +218 -0
  79. package/backend/src/flowent/prompts/__init__.py +67 -0
  80. package/backend/src/flowent/prompts/__pycache__/__init__.cpython-313.pyc +0 -0
  81. package/backend/src/flowent/prompts/__pycache__/common.cpython-313.pyc +0 -0
  82. package/backend/src/flowent/prompts/__pycache__/steward.cpython-313.pyc +0 -0
  83. package/backend/src/flowent/prompts/common.py +250 -0
  84. package/backend/src/flowent/prompts/steward.py +64 -0
  85. package/backend/src/flowent/providers/__init__.py +23 -0
  86. package/backend/src/flowent/providers/__pycache__/__init__.cpython-313.pyc +0 -0
  87. package/backend/src/flowent/providers/__pycache__/anthropic.cpython-313.pyc +0 -0
  88. package/backend/src/flowent/providers/__pycache__/base_url.cpython-313.pyc +0 -0
  89. package/backend/src/flowent/providers/__pycache__/configuration.cpython-313.pyc +0 -0
  90. package/backend/src/flowent/providers/__pycache__/content.cpython-313.pyc +0 -0
  91. package/backend/src/flowent/providers/__pycache__/errors.cpython-313.pyc +0 -0
  92. package/backend/src/flowent/providers/__pycache__/gateway.cpython-313.pyc +0 -0
  93. package/backend/src/flowent/providers/__pycache__/headers.cpython-313.pyc +0 -0
  94. package/backend/src/flowent/providers/__pycache__/management.cpython-313.pyc +0 -0
  95. package/backend/src/flowent/providers/__pycache__/openai.cpython-313.pyc +0 -0
  96. package/backend/src/flowent/providers/__pycache__/openai_responses.cpython-313.pyc +0 -0
  97. package/backend/src/flowent/providers/__pycache__/registry.cpython-313.pyc +0 -0
  98. package/backend/src/flowent/providers/__pycache__/sse.cpython-313.pyc +0 -0
  99. package/backend/src/flowent/providers/__pycache__/thinking.cpython-313.pyc +0 -0
  100. package/backend/src/flowent/providers/anthropic.py +468 -0
  101. package/backend/src/flowent/providers/base_url.py +60 -0
  102. package/backend/src/flowent/providers/configuration.py +189 -0
  103. package/backend/src/flowent/providers/content.py +122 -0
  104. package/backend/src/flowent/providers/errors.py +223 -0
  105. package/backend/src/flowent/providers/gateway.py +169 -0
  106. package/backend/src/flowent/providers/gemini.py +447 -0
  107. package/backend/src/flowent/providers/headers.py +20 -0
  108. package/backend/src/flowent/providers/management.py +96 -0
  109. package/backend/src/flowent/providers/ollama.py +293 -0
  110. package/backend/src/flowent/providers/openai.py +422 -0
  111. package/backend/src/flowent/providers/openai_responses.py +655 -0
  112. package/backend/src/flowent/providers/registry.py +144 -0
  113. package/backend/src/flowent/providers/sse.py +31 -0
  114. package/backend/src/flowent/providers/thinking.py +79 -0
  115. package/backend/src/flowent/registry.py +73 -0
  116. package/backend/src/flowent/role_management.py +267 -0
  117. package/backend/src/flowent/routes/__init__.py +28 -0
  118. package/backend/src/flowent/routes/__pycache__/__init__.cpython-313.pyc +0 -0
  119. package/backend/src/flowent/routes/__pycache__/access.cpython-313.pyc +0 -0
  120. package/backend/src/flowent/routes/__pycache__/assistant.cpython-313.pyc +0 -0
  121. package/backend/src/flowent/routes/__pycache__/image_assets.cpython-313.pyc +0 -0
  122. package/backend/src/flowent/routes/__pycache__/mcp.cpython-313.pyc +0 -0
  123. package/backend/src/flowent/routes/__pycache__/meta.cpython-313.pyc +0 -0
  124. package/backend/src/flowent/routes/__pycache__/nodes.cpython-313.pyc +0 -0
  125. package/backend/src/flowent/routes/__pycache__/prompts.cpython-313.pyc +0 -0
  126. package/backend/src/flowent/routes/__pycache__/providers_route.cpython-313.pyc +0 -0
  127. package/backend/src/flowent/routes/__pycache__/roles.cpython-313.pyc +0 -0
  128. package/backend/src/flowent/routes/__pycache__/settings.cpython-313.pyc +0 -0
  129. package/backend/src/flowent/routes/__pycache__/tabs.cpython-313.pyc +0 -0
  130. package/backend/src/flowent/routes/__pycache__/ws.cpython-313.pyc +0 -0
  131. package/backend/src/flowent/routes/access.py +48 -0
  132. package/backend/src/flowent/routes/assistant.py +155 -0
  133. package/backend/src/flowent/routes/image_assets.py +33 -0
  134. package/backend/src/flowent/routes/mcp.py +125 -0
  135. package/backend/src/flowent/routes/meta.py +28 -0
  136. package/backend/src/flowent/routes/nodes.py +413 -0
  137. package/backend/src/flowent/routes/prompts.py +46 -0
  138. package/backend/src/flowent/routes/providers_route.py +365 -0
  139. package/backend/src/flowent/routes/roles.py +207 -0
  140. package/backend/src/flowent/routes/settings.py +328 -0
  141. package/backend/src/flowent/routes/tabs.py +310 -0
  142. package/backend/src/flowent/routes/ws.py +33 -0
  143. package/backend/src/flowent/runtime.py +165 -0
  144. package/backend/src/flowent/sandbox.py +45 -0
  145. package/backend/src/flowent/security.py +57 -0
  146. package/backend/src/flowent/settings.py +2518 -0
  147. package/backend/src/flowent/settings_management.py +298 -0
  148. package/backend/src/flowent/state_db.py +120 -0
  149. package/backend/src/flowent/static/assets/AssistantPage-VBohhz4d.js +1 -0
  150. package/backend/src/flowent/static/assets/ChannelsPage-CIydPZA_.js +1 -0
  151. package/backend/src/flowent/static/assets/McpPage-CHPm2TPY.js +7 -0
  152. package/backend/src/flowent/static/assets/PageScaffold-DteOA8V7.js +1 -0
  153. package/backend/src/flowent/static/assets/PromptsPage-CSmJ3sZg.js +1 -0
  154. package/backend/src/flowent/static/assets/ProvidersPage-sl2jeG4e.js +3 -0
  155. package/backend/src/flowent/static/assets/RolesPage-DCe7W6Km.js +1 -0
  156. package/backend/src/flowent/static/assets/SettingsPage-Bix9e63E.js +3 -0
  157. package/backend/src/flowent/static/assets/ToolsPage-favNkj5C.js +1 -0
  158. package/backend/src/flowent/static/assets/WorkspaceCommandDialog-DRS6wiD6.js +1 -0
  159. package/backend/src/flowent/static/assets/WorkspacePage-KuaDjt_D.js +3 -0
  160. package/backend/src/flowent/static/assets/WorkspacePanels-BZxBw8M5.js +1 -0
  161. package/backend/src/flowent/static/assets/alert-dialog-DIBUCmqM.js +1 -0
  162. package/backend/src/flowent/static/assets/datetime-eJqd0V2S.js +1 -0
  163. package/backend/src/flowent/static/assets/dialog-BOvHIBrg.js +1 -0
  164. package/backend/src/flowent/static/assets/elk-worker.min-C9JGDOE-.js +6312 -0
  165. package/backend/src/flowent/static/assets/graph-vendor-CHpVij2M.css +1 -0
  166. package/backend/src/flowent/static/assets/graph-vendor-DRq_-6fV.js +7 -0
  167. package/backend/src/flowent/static/assets/index-Biio-CoI.js +10 -0
  168. package/backend/src/flowent/static/assets/index-CmQvO7sl.css +1 -0
  169. package/backend/src/flowent/static/assets/layout.worker-jMHqAFbP.js +24 -0
  170. package/backend/src/flowent/static/assets/markdown-vendor-C9RtvaJh.js +29 -0
  171. package/backend/src/flowent/static/assets/modelParams-DcEhGnu0.js +1 -0
  172. package/backend/src/flowent/static/assets/react-vendor-mEs_JJxa.js +9 -0
  173. package/backend/src/flowent/static/assets/roles-BbIEIMeG.js +1 -0
  174. package/backend/src/flowent/static/assets/rolldown-runtime-BYbx6iT9.js +1 -0
  175. package/backend/src/flowent/static/assets/select-D9SwnlXF.js +1 -0
  176. package/backend/src/flowent/static/assets/surface-Bzr1FRG4.js +1 -0
  177. package/backend/src/flowent/static/assets/triState-DgLlKdRR.js +1 -0
  178. package/backend/src/flowent/static/assets/ui-vendor-UazN8rcv.js +51 -0
  179. package/backend/src/flowent/static/index.html +35 -0
  180. package/backend/src/flowent/tools/__init__.py +275 -0
  181. package/backend/src/flowent/tools/__pycache__/__init__.cpython-313.pyc +0 -0
  182. package/backend/src/flowent/tools/__pycache__/connect.cpython-313.pyc +0 -0
  183. package/backend/src/flowent/tools/__pycache__/contacts.cpython-313.pyc +0 -0
  184. package/backend/src/flowent/tools/__pycache__/create_agent.cpython-313.pyc +0 -0
  185. package/backend/src/flowent/tools/__pycache__/create_tab.cpython-313.pyc +0 -0
  186. package/backend/src/flowent/tools/__pycache__/delete_tab.cpython-313.pyc +0 -0
  187. package/backend/src/flowent/tools/__pycache__/edit.cpython-313.pyc +0 -0
  188. package/backend/src/flowent/tools/__pycache__/exec.cpython-313.pyc +0 -0
  189. package/backend/src/flowent/tools/__pycache__/fetch.cpython-313.pyc +0 -0
  190. package/backend/src/flowent/tools/__pycache__/idle.cpython-313.pyc +0 -0
  191. package/backend/src/flowent/tools/__pycache__/list_roles.cpython-313.pyc +0 -0
  192. package/backend/src/flowent/tools/__pycache__/list_tabs.cpython-313.pyc +0 -0
  193. package/backend/src/flowent/tools/__pycache__/list_tools.cpython-313.pyc +0 -0
  194. package/backend/src/flowent/tools/__pycache__/manage_prompts.cpython-313.pyc +0 -0
  195. package/backend/src/flowent/tools/__pycache__/manage_providers.cpython-313.pyc +0 -0
  196. package/backend/src/flowent/tools/__pycache__/manage_roles.cpython-313.pyc +0 -0
  197. package/backend/src/flowent/tools/__pycache__/manage_settings.cpython-313.pyc +0 -0
  198. package/backend/src/flowent/tools/__pycache__/mcp.cpython-313.pyc +0 -0
  199. package/backend/src/flowent/tools/__pycache__/read.cpython-313.pyc +0 -0
  200. package/backend/src/flowent/tools/__pycache__/send.cpython-313.pyc +0 -0
  201. package/backend/src/flowent/tools/__pycache__/set_permissions.cpython-313.pyc +0 -0
  202. package/backend/src/flowent/tools/__pycache__/sleep.cpython-313.pyc +0 -0
  203. package/backend/src/flowent/tools/__pycache__/todo.cpython-313.pyc +0 -0
  204. package/backend/src/flowent/tools/connect.py +100 -0
  205. package/backend/src/flowent/tools/contacts.py +22 -0
  206. package/backend/src/flowent/tools/create_agent.py +191 -0
  207. package/backend/src/flowent/tools/create_tab.py +61 -0
  208. package/backend/src/flowent/tools/delete_tab.py +39 -0
  209. package/backend/src/flowent/tools/edit.py +142 -0
  210. package/backend/src/flowent/tools/exec.py +118 -0
  211. package/backend/src/flowent/tools/fetch.py +85 -0
  212. package/backend/src/flowent/tools/idle.py +27 -0
  213. package/backend/src/flowent/tools/list_roles.py +75 -0
  214. package/backend/src/flowent/tools/list_tabs.py +100 -0
  215. package/backend/src/flowent/tools/list_tools.py +28 -0
  216. package/backend/src/flowent/tools/manage_prompts.py +102 -0
  217. package/backend/src/flowent/tools/manage_providers.py +220 -0
  218. package/backend/src/flowent/tools/manage_roles.py +275 -0
  219. package/backend/src/flowent/tools/manage_settings.py +364 -0
  220. package/backend/src/flowent/tools/mcp.py +199 -0
  221. package/backend/src/flowent/tools/read.py +152 -0
  222. package/backend/src/flowent/tools/send.py +68 -0
  223. package/backend/src/flowent/tools/set_permissions.py +99 -0
  224. package/backend/src/flowent/tools/sleep.py +41 -0
  225. package/backend/src/flowent/tools/todo.py +51 -0
  226. package/backend/src/flowent/workspace_store.py +479 -0
  227. package/backend/tests/__init__.py +0 -0
  228. package/backend/tests/__pycache__/__init__.cpython-313.pyc +0 -0
  229. package/backend/tests/__pycache__/conftest.cpython-313-pytest-9.0.3.pyc +0 -0
  230. package/backend/tests/conftest.py +6 -0
  231. package/backend/tests/integration/api/__pycache__/conftest.cpython-313-pytest-9.0.3.pyc +0 -0
  232. package/backend/tests/integration/api/__pycache__/test_access_api.cpython-313-pytest-9.0.3.pyc +0 -0
  233. package/backend/tests/integration/api/__pycache__/test_assistant_api.cpython-313-pytest-9.0.3.pyc +0 -0
  234. package/backend/tests/integration/api/__pycache__/test_frontend_mounting.cpython-313-pytest-9.0.3.pyc +0 -0
  235. package/backend/tests/integration/api/__pycache__/test_mcp_api.cpython-313-pytest-9.0.3.pyc +0 -0
  236. package/backend/tests/integration/api/__pycache__/test_meta_api.cpython-313-pytest-9.0.3.pyc +0 -0
  237. package/backend/tests/integration/api/__pycache__/test_nodes_api.cpython-313-pytest-9.0.3.pyc +0 -0
  238. package/backend/tests/integration/api/__pycache__/test_prompts_api.cpython-313-pytest-9.0.3.pyc +0 -0
  239. package/backend/tests/integration/api/__pycache__/test_roles_api.cpython-313-pytest-9.0.3.pyc +0 -0
  240. package/backend/tests/integration/api/__pycache__/test_tabs_api.cpython-313-pytest-9.0.3.pyc +0 -0
  241. package/backend/tests/integration/api/conftest.py +29 -0
  242. package/backend/tests/integration/api/test_access_api.py +182 -0
  243. package/backend/tests/integration/api/test_assistant_api.py +354 -0
  244. package/backend/tests/integration/api/test_frontend_mounting.py +61 -0
  245. package/backend/tests/integration/api/test_mcp_api.py +116 -0
  246. package/backend/tests/integration/api/test_meta_api.py +33 -0
  247. package/backend/tests/integration/api/test_nodes_api.py +722 -0
  248. package/backend/tests/integration/api/test_prompts_api.py +47 -0
  249. package/backend/tests/integration/api/test_roles_api.py +228 -0
  250. package/backend/tests/integration/api/test_tabs_api.py +802 -0
  251. package/backend/tests/unit/__pycache__/test_access.cpython-313-pytest-9.0.3.pyc +0 -0
  252. package/backend/tests/unit/__pycache__/test_cli.cpython-313-pytest-9.0.3.pyc +0 -0
  253. package/backend/tests/unit/__pycache__/test_graph_runtime.cpython-313-pytest-9.0.3.pyc +0 -0
  254. package/backend/tests/unit/__pycache__/test_network.cpython-313-pytest-9.0.3.pyc +0 -0
  255. package/backend/tests/unit/__pycache__/test_state_sqlite_storage.cpython-313-pytest-9.0.3.pyc +0 -0
  256. package/backend/tests/unit/__pycache__/test_workspace_store.cpython-313-pytest-9.0.3.pyc +0 -0
  257. package/backend/tests/unit/agent/__pycache__/test_agent_public_api.cpython-313-pytest-9.0.3.pyc +0 -0
  258. package/backend/tests/unit/agent/__pycache__/test_agent_runtime.cpython-313-pytest-9.0.3.pyc +0 -0
  259. package/backend/tests/unit/agent/test_agent_public_api.py +837 -0
  260. package/backend/tests/unit/agent/test_agent_runtime.py +2942 -0
  261. package/backend/tests/unit/channels/__pycache__/test_telegram_channel.cpython-313-pytest-9.0.3.pyc +0 -0
  262. package/backend/tests/unit/channels/test_telegram_channel.py +552 -0
  263. package/backend/tests/unit/logging/__pycache__/test_logging.cpython-313-pytest-9.0.3.pyc +0 -0
  264. package/backend/tests/unit/logging/test_logging.py +132 -0
  265. package/backend/tests/unit/prompts/__pycache__/test_prompts.cpython-313-pytest-9.0.3.pyc +0 -0
  266. package/backend/tests/unit/prompts/test_prompts.py +570 -0
  267. package/backend/tests/unit/providers/__pycache__/test_anthropic_provider.cpython-313-pytest-9.0.3.pyc +0 -0
  268. package/backend/tests/unit/providers/__pycache__/test_errors.cpython-313-pytest-9.0.3.pyc +0 -0
  269. package/backend/tests/unit/providers/__pycache__/test_extract_delta_parts.cpython-313-pytest-9.0.3.pyc +0 -0
  270. package/backend/tests/unit/providers/__pycache__/test_openai_provider.cpython-313-pytest-9.0.3.pyc +0 -0
  271. package/backend/tests/unit/providers/__pycache__/test_openai_responses.cpython-313-pytest-9.0.3.pyc +0 -0
  272. package/backend/tests/unit/providers/__pycache__/test_provider_gateway.cpython-313-pytest-9.0.3.pyc +0 -0
  273. package/backend/tests/unit/providers/__pycache__/test_think_tag_parser.cpython-313-pytest-9.0.3.pyc +0 -0
  274. package/backend/tests/unit/providers/test_anthropic_provider.py +185 -0
  275. package/backend/tests/unit/providers/test_errors.py +68 -0
  276. package/backend/tests/unit/providers/test_extract_delta_parts.py +22 -0
  277. package/backend/tests/unit/providers/test_openai_provider.py +139 -0
  278. package/backend/tests/unit/providers/test_openai_responses.py +402 -0
  279. package/backend/tests/unit/providers/test_provider_gateway.py +359 -0
  280. package/backend/tests/unit/providers/test_think_tag_parser.py +36 -0
  281. package/backend/tests/unit/routes/__pycache__/test_prompts_routes.cpython-313-pytest-9.0.3.pyc +0 -0
  282. package/backend/tests/unit/routes/__pycache__/test_providers_route.cpython-313-pytest-9.0.3.pyc +0 -0
  283. package/backend/tests/unit/routes/__pycache__/test_roles_routes.cpython-313-pytest-9.0.3.pyc +0 -0
  284. package/backend/tests/unit/routes/__pycache__/test_settings_routes.cpython-313-pytest-9.0.3.pyc +0 -0
  285. package/backend/tests/unit/routes/test_prompts_routes.py +104 -0
  286. package/backend/tests/unit/routes/test_providers_route.py +370 -0
  287. package/backend/tests/unit/routes/test_roles_routes.py +535 -0
  288. package/backend/tests/unit/routes/test_settings_routes.py +1142 -0
  289. package/backend/tests/unit/runtime/__pycache__/test_bootstrap_runtime.cpython-313-pytest-9.0.3.pyc +0 -0
  290. package/backend/tests/unit/runtime/test_bootstrap_runtime.py +1002 -0
  291. package/backend/tests/unit/sandbox/__pycache__/test_sandbox_tools.cpython-313-pytest-9.0.3.pyc +0 -0
  292. package/backend/tests/unit/sandbox/test_sandbox_tools.py +78 -0
  293. package/backend/tests/unit/security/__pycache__/test_security.cpython-313-pytest-9.0.3.pyc +0 -0
  294. package/backend/tests/unit/security/test_security.py +124 -0
  295. package/backend/tests/unit/settings/__pycache__/test_settings_roles.cpython-313-pytest-9.0.3.pyc +0 -0
  296. package/backend/tests/unit/settings/test_settings_roles.py +751 -0
  297. package/backend/tests/unit/test_access.py +45 -0
  298. package/backend/tests/unit/test_cli.py +124 -0
  299. package/backend/tests/unit/test_graph_runtime.py +72 -0
  300. package/backend/tests/unit/test_network.py +51 -0
  301. package/backend/tests/unit/test_state_sqlite_storage.py +159 -0
  302. package/backend/tests/unit/test_workspace_store.py +231 -0
  303. package/backend/tests/unit/tools/__pycache__/test_connect_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  304. package/backend/tests/unit/tools/__pycache__/test_create_agent_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  305. package/backend/tests/unit/tools/__pycache__/test_delete_tab_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  306. package/backend/tests/unit/tools/__pycache__/test_edit_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  307. package/backend/tests/unit/tools/__pycache__/test_exec_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  308. package/backend/tests/unit/tools/__pycache__/test_fetch_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  309. package/backend/tests/unit/tools/__pycache__/test_manage_prompts_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  310. package/backend/tests/unit/tools/__pycache__/test_manage_providers_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  311. package/backend/tests/unit/tools/__pycache__/test_manage_roles_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  312. package/backend/tests/unit/tools/__pycache__/test_manage_settings_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  313. package/backend/tests/unit/tools/__pycache__/test_read_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  314. package/backend/tests/unit/tools/__pycache__/test_set_permissions_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  315. package/backend/tests/unit/tools/__pycache__/test_todo_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  316. package/backend/tests/unit/tools/__pycache__/test_tool_registry.cpython-313-pytest-9.0.3.pyc +0 -0
  317. package/backend/tests/unit/tools/test_connect_tool.py +228 -0
  318. package/backend/tests/unit/tools/test_create_agent_tool.py +436 -0
  319. package/backend/tests/unit/tools/test_delete_tab_tool.py +116 -0
  320. package/backend/tests/unit/tools/test_edit_tool.py +115 -0
  321. package/backend/tests/unit/tools/test_exec_tool.py +81 -0
  322. package/backend/tests/unit/tools/test_fetch_tool.py +65 -0
  323. package/backend/tests/unit/tools/test_manage_prompts_tool.py +117 -0
  324. package/backend/tests/unit/tools/test_manage_providers_tool.py +460 -0
  325. package/backend/tests/unit/tools/test_manage_roles_tool.py +411 -0
  326. package/backend/tests/unit/tools/test_manage_settings_tool.py +611 -0
  327. package/backend/tests/unit/tools/test_read_tool.py +33 -0
  328. package/backend/tests/unit/tools/test_set_permissions_tool.py +595 -0
  329. package/backend/tests/unit/tools/test_todo_tool.py +37 -0
  330. package/backend/tests/unit/tools/test_tool_registry.py +194 -0
  331. package/backend/uv.lock +1144 -0
  332. package/bin/flowent.mjs +62 -36
  333. package/dist/frontend/assets/AssistantPage-VBohhz4d.js +1 -0
  334. package/dist/frontend/assets/ChannelsPage-CIydPZA_.js +1 -0
  335. package/dist/frontend/assets/McpPage-CHPm2TPY.js +7 -0
  336. package/dist/frontend/assets/PageScaffold-DteOA8V7.js +1 -0
  337. package/dist/frontend/assets/PromptsPage-CSmJ3sZg.js +1 -0
  338. package/dist/frontend/assets/ProvidersPage-sl2jeG4e.js +3 -0
  339. package/dist/frontend/assets/RolesPage-DCe7W6Km.js +1 -0
  340. package/dist/frontend/assets/SettingsPage-Bix9e63E.js +3 -0
  341. package/dist/frontend/assets/ToolsPage-favNkj5C.js +1 -0
  342. package/dist/frontend/assets/WorkspaceCommandDialog-DRS6wiD6.js +1 -0
  343. package/dist/frontend/assets/WorkspacePage-KuaDjt_D.js +3 -0
  344. package/dist/frontend/assets/WorkspacePanels-BZxBw8M5.js +1 -0
  345. package/dist/frontend/assets/alert-dialog-DIBUCmqM.js +1 -0
  346. package/dist/frontend/assets/datetime-eJqd0V2S.js +1 -0
  347. package/dist/frontend/assets/dialog-BOvHIBrg.js +1 -0
  348. package/dist/frontend/assets/elk-worker.min-C9JGDOE-.js +6312 -0
  349. package/dist/frontend/assets/graph-vendor-CHpVij2M.css +1 -0
  350. package/dist/frontend/assets/graph-vendor-DRq_-6fV.js +7 -0
  351. package/dist/frontend/assets/index-Biio-CoI.js +10 -0
  352. package/dist/frontend/assets/index-CmQvO7sl.css +1 -0
  353. package/dist/frontend/assets/layout.worker-jMHqAFbP.js +24 -0
  354. package/dist/frontend/assets/markdown-vendor-C9RtvaJh.js +29 -0
  355. package/dist/frontend/assets/modelParams-DcEhGnu0.js +1 -0
  356. package/dist/frontend/assets/react-vendor-mEs_JJxa.js +9 -0
  357. package/dist/frontend/assets/roles-BbIEIMeG.js +1 -0
  358. package/dist/frontend/assets/rolldown-runtime-BYbx6iT9.js +1 -0
  359. package/dist/frontend/assets/select-D9SwnlXF.js +1 -0
  360. package/dist/frontend/assets/surface-Bzr1FRG4.js +1 -0
  361. package/dist/frontend/assets/triState-DgLlKdRR.js +1 -0
  362. package/dist/frontend/assets/ui-vendor-UazN8rcv.js +51 -0
  363. package/dist/frontend/index.html +35 -0
  364. package/package.json +27 -41
  365. package/dist/.next/BUILD_ID +0 -1
  366. package/dist/.next/app-path-routes-manifest.json +0 -6
  367. package/dist/.next/build-manifest.json +0 -20
  368. package/dist/.next/package.json +0 -1
  369. package/dist/.next/prerender-manifest.json +0 -114
  370. package/dist/.next/required-server-files.json +0 -333
  371. package/dist/.next/routes-manifest.json +0 -69
  372. package/dist/.next/server/app/_global-error/page/app-paths-manifest.json +0 -3
  373. package/dist/.next/server/app/_global-error/page/build-manifest.json +0 -16
  374. package/dist/.next/server/app/_global-error/page/next-font-manifest.json +0 -6
  375. package/dist/.next/server/app/_global-error/page/react-loadable-manifest.json +0 -1
  376. package/dist/.next/server/app/_global-error/page/server-reference-manifest.json +0 -4
  377. package/dist/.next/server/app/_global-error/page.js +0 -9
  378. package/dist/.next/server/app/_global-error/page.js.map +0 -5
  379. package/dist/.next/server/app/_global-error/page.js.nft.json +0 -1
  380. package/dist/.next/server/app/_global-error/page_client-reference-manifest.js +0 -3
  381. package/dist/.next/server/app/_global-error.html +0 -1
  382. package/dist/.next/server/app/_global-error.meta +0 -15
  383. package/dist/.next/server/app/_global-error.rsc +0 -14
  384. package/dist/.next/server/app/_global-error.segments/__PAGE__.segment.rsc +0 -5
  385. package/dist/.next/server/app/_global-error.segments/_full.segment.rsc +0 -14
  386. package/dist/.next/server/app/_global-error.segments/_head.segment.rsc +0 -5
  387. package/dist/.next/server/app/_global-error.segments/_index.segment.rsc +0 -5
  388. package/dist/.next/server/app/_global-error.segments/_tree.segment.rsc +0 -1
  389. package/dist/.next/server/app/_not-found/page/app-paths-manifest.json +0 -3
  390. package/dist/.next/server/app/_not-found/page/build-manifest.json +0 -16
  391. package/dist/.next/server/app/_not-found/page/next-font-manifest.json +0 -10
  392. package/dist/.next/server/app/_not-found/page/react-loadable-manifest.json +0 -1
  393. package/dist/.next/server/app/_not-found/page/server-reference-manifest.json +0 -4
  394. package/dist/.next/server/app/_not-found/page.js +0 -13
  395. package/dist/.next/server/app/_not-found/page.js.map +0 -5
  396. package/dist/.next/server/app/_not-found/page.js.nft.json +0 -1
  397. package/dist/.next/server/app/_not-found/page_client-reference-manifest.js +0 -3
  398. package/dist/.next/server/app/_not-found.html +0 -1
  399. package/dist/.next/server/app/_not-found.meta +0 -16
  400. package/dist/.next/server/app/_not-found.rsc +0 -16
  401. package/dist/.next/server/app/_not-found.segments/_full.segment.rsc +0 -16
  402. package/dist/.next/server/app/_not-found.segments/_head.segment.rsc +0 -6
  403. package/dist/.next/server/app/_not-found.segments/_index.segment.rsc +0 -5
  404. package/dist/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +0 -5
  405. package/dist/.next/server/app/_not-found.segments/_not-found.segment.rsc +0 -5
  406. package/dist/.next/server/app/_not-found.segments/_tree.segment.rsc +0 -2
  407. package/dist/.next/server/app/icon.svg/route/app-paths-manifest.json +0 -3
  408. package/dist/.next/server/app/icon.svg/route/build-manifest.json +0 -9
  409. package/dist/.next/server/app/icon.svg/route.js +0 -6
  410. package/dist/.next/server/app/icon.svg/route.js.map +0 -5
  411. package/dist/.next/server/app/icon.svg/route.js.nft.json +0 -1
  412. package/dist/.next/server/app/icon.svg.meta +0 -1
  413. package/dist/.next/server/app/index.html +0 -1
  414. package/dist/.next/server/app/index.meta +0 -14
  415. package/dist/.next/server/app/index.rsc +0 -15
  416. package/dist/.next/server/app/index.segments/__PAGE__.segment.rsc +0 -5
  417. package/dist/.next/server/app/index.segments/_full.segment.rsc +0 -15
  418. package/dist/.next/server/app/index.segments/_head.segment.rsc +0 -6
  419. package/dist/.next/server/app/index.segments/_index.segment.rsc +0 -5
  420. package/dist/.next/server/app/index.segments/_tree.segment.rsc +0 -3
  421. package/dist/.next/server/app/page/app-paths-manifest.json +0 -3
  422. package/dist/.next/server/app/page/build-manifest.json +0 -16
  423. package/dist/.next/server/app/page/next-font-manifest.json +0 -10
  424. package/dist/.next/server/app/page/react-loadable-manifest.json +0 -1
  425. package/dist/.next/server/app/page/server-reference-manifest.json +0 -4
  426. package/dist/.next/server/app/page.js +0 -14
  427. package/dist/.next/server/app/page.js.map +0 -5
  428. package/dist/.next/server/app/page.js.nft.json +0 -1
  429. package/dist/.next/server/app/page_client-reference-manifest.js +0 -3
  430. package/dist/.next/server/app-paths-manifest.json +0 -6
  431. package/dist/.next/server/chunks/[externals]_next_dist_0arv.vj._.js +0 -3
  432. package/dist/.next/server/chunks/[root-of-the-server]__0vcj1q1._.js +0 -13
  433. package/dist/.next/server/chunks/[turbopack]_runtime.js +0 -903
  434. package/dist/.next/server/chunks/_next-internal_server_app_icon_svg_route_actions_0-0ehc~.js +0 -3
  435. package/dist/.next/server/chunks/ssr/05w9_next_dist_0ihu0u9._.js +0 -6
  436. package/dist/.next/server/chunks/ssr/05w9_next_dist_client_components_12u3mib._.js +0 -3
  437. package/dist/.next/server/chunks/ssr/05w9_next_dist_client_components_builtin_forbidden_04fbe_..js +0 -3
  438. package/dist/.next/server/chunks/ssr/05w9_next_dist_client_components_builtin_global-error_0brpl_..js +0 -3
  439. package/dist/.next/server/chunks/ssr/05w9_next_dist_client_components_builtin_unauthorized_0~2g66g.js +0 -3
  440. package/dist/.next/server/chunks/ssr/05w9_next_dist_esm_build_templates_app-page_0~cyr1_.js +0 -4
  441. package/dist/.next/server/chunks/ssr/05w9_next_dist_esm_build_templates_app-page_1105emf.js +0 -4
  442. package/dist/.next/server/chunks/ssr/05w9_next_dist_esm_build_templates_app-page_11uhyqv.js +0 -4
  443. package/dist/.next/server/chunks/ssr/[root-of-the-server]__0.t9_75._.js +0 -33
  444. package/dist/.next/server/chunks/ssr/[root-of-the-server]__0c0ud_z._.js +0 -3
  445. package/dist/.next/server/chunks/ssr/[root-of-the-server]__0f9_8d4._.js +0 -3
  446. package/dist/.next/server/chunks/ssr/[root-of-the-server]__0l5ko41._.js +0 -19
  447. package/dist/.next/server/chunks/ssr/[root-of-the-server]__0mn6z7i._.js +0 -3
  448. package/dist/.next/server/chunks/ssr/[root-of-the-server]__0npxxst._.js +0 -33
  449. package/dist/.next/server/chunks/ssr/[root-of-the-server]__0qjhaca._.js +0 -3
  450. package/dist/.next/server/chunks/ssr/[root-of-the-server]__0rwgw3s._.js +0 -3
  451. package/dist/.next/server/chunks/ssr/[turbopack]_runtime.js +0 -903
  452. package/dist/.next/server/chunks/ssr/_next-internal_server_app__global-error_page_actions_0k77kol.js +0 -3
  453. package/dist/.next/server/chunks/ssr/_next-internal_server_app__not-found_page_actions_0eq97pa.js +0 -3
  454. package/dist/.next/server/chunks/ssr/_next-internal_server_app_page_actions_09-gtaw.js +0 -3
  455. package/dist/.next/server/chunks/ssr/node_modules__pnpm_056~6.6._.js +0 -3
  456. package/dist/.next/server/chunks/ssr/node_modules__pnpm_0~j0k.e._.js +0 -33
  457. package/dist/.next/server/functions-config-manifest.json +0 -4
  458. package/dist/.next/server/middleware-build-manifest.js +0 -20
  459. package/dist/.next/server/middleware-manifest.json +0 -6
  460. package/dist/.next/server/next-font-manifest.js +0 -1
  461. package/dist/.next/server/next-font-manifest.json +0 -13
  462. package/dist/.next/server/pages/404.html +0 -1
  463. package/dist/.next/server/pages/500.html +0 -1
  464. package/dist/.next/server/pages-manifest.json +0 -4
  465. package/dist/.next/server/prefetch-hints.json +0 -1
  466. package/dist/.next/server/server-reference-manifest.js +0 -1
  467. package/dist/.next/server/server-reference-manifest.json +0 -5
  468. package/dist/.next/static/SMWpxFVvkpYFxY7uuFvGB/_buildManifest.js +0 -11
  469. package/dist/.next/static/SMWpxFVvkpYFxY7uuFvGB/_clientMiddlewareManifest.js +0 -1
  470. package/dist/.next/static/SMWpxFVvkpYFxY7uuFvGB/_ssgManifest.js +0 -1
  471. package/dist/.next/static/chunks/01qk2~bgf76vu.js +0 -1
  472. package/dist/.next/static/chunks/03~yq9q893hmn.js +0 -1
  473. package/dist/.next/static/chunks/080queev.r2uy.js +0 -31
  474. package/dist/.next/static/chunks/0v3lyuj75aq50.js +0 -1
  475. package/dist/.next/static/chunks/10b~xdx5c-i7s.js +0 -5
  476. package/dist/.next/static/chunks/14gla2ascffgv.css +0 -2
  477. package/dist/.next/static/chunks/turbopack-0m-970~qvs7sc.js +0 -1
  478. package/dist/.next/static/media/7178b3e590c64307-s.11.cyxs5p-0z~.woff2 +0 -0
  479. package/dist/.next/static/media/8a480f0b521d4e75-s.06d3mdzz5bre_.woff2 +0 -0
  480. package/dist/.next/static/media/caa3a2e1cccd8315-s.p.16t1db8_9y2o~.woff2 +0 -0
  481. package/dist/package.json +0 -88
  482. package/dist/server.js +0 -38
  483. /package/{dist/.next/server/app/icon.svg.body → backend/src/flowent/static/favicon.svg} +0 -0
  484. /package/dist/{.next/static/media/icon.0.r~afrtrocz9.svg → frontend/favicon.svg} +0 -0
@@ -0,0 +1,802 @@
1
+ from copy import deepcopy
2
+ from typing import Any
3
+
4
+ from fastapi.testclient import TestClient
5
+
6
+
7
+ def _create_agent_node(
8
+ client: TestClient,
9
+ *,
10
+ tab_id: str,
11
+ name: str,
12
+ role_name: str = "Worker",
13
+ ) -> dict[str, Any]:
14
+ response = client.post(
15
+ f"/api/workflows/{tab_id}/nodes",
16
+ json={"role_name": role_name, "name": name},
17
+ )
18
+ assert response.status_code == 200
19
+ return response.json()
20
+
21
+
22
+ def _create_graph_node(
23
+ client: TestClient,
24
+ *,
25
+ tab_id: str,
26
+ node_type: str,
27
+ name: str,
28
+ config: dict[str, object] | None = None,
29
+ ) -> dict[str, Any]:
30
+ response = client.post(
31
+ f"/api/workflows/{tab_id}/nodes",
32
+ json={
33
+ "node_type": node_type,
34
+ "name": name,
35
+ "config": config or {},
36
+ },
37
+ )
38
+ assert response.status_code == 200
39
+ return response.json()
40
+
41
+
42
+ def test_list_tabs_is_empty_at_startup(client: TestClient):
43
+ response = client.get("/api/workflows")
44
+
45
+ assert response.status_code == 200
46
+ assert response.json() == {"workflows": []}
47
+
48
+
49
+ def test_create_tab_rejects_removed_mcp_servers_field(client: TestClient):
50
+ response = client.post(
51
+ "/api/workflows",
52
+ json={"title": "Review Task", "mcp_servers": ["filesystem"]},
53
+ )
54
+
55
+ assert response.status_code == 422
56
+
57
+
58
+ def test_create_tab_rejects_removed_goal_field(client: TestClient):
59
+ response = client.post(
60
+ "/api/workflows",
61
+ json={"title": "Review Task", "goal": "Inspect changed files"},
62
+ )
63
+
64
+ assert response.status_code == 422
65
+
66
+
67
+ def test_create_tab_node_and_edge_round_trip(client: TestClient):
68
+ create_tab_response = client.post(
69
+ "/api/workflows",
70
+ json={
71
+ "title": "Review Task",
72
+ "allow_network": True,
73
+ "write_dirs": ["/tmp"],
74
+ },
75
+ )
76
+
77
+ assert create_tab_response.status_code == 200
78
+ tab = create_tab_response.json()
79
+ tab_id = tab["id"]
80
+ assert tab["title"] == "Review Task"
81
+ assert "goal" not in tab
82
+ assert tab["node_count"] == 0
83
+ assert tab["edge_count"] == 0
84
+ assert tab["activation_state"] == "inactive"
85
+ assert tab["allow_network"] is True
86
+ assert tab["write_dirs"] == ["/tmp"]
87
+ assert tab["definition"] == {"version": 1, "nodes": [], "edges": []}
88
+ assert isinstance(tab["leader_id"], str)
89
+
90
+ reader = _create_agent_node(client, tab_id=tab_id, name="Reader")
91
+ writer = _create_agent_node(client, tab_id=tab_id, name="Writer")
92
+
93
+ assert reader["workflow_id"] == tab_id
94
+ assert writer["workflow_id"] == tab_id
95
+ assert reader["node_type"] == "agent"
96
+ assert writer["node_type"] == "agent"
97
+ assert reader["config"]["role_name"] == "Worker"
98
+ assert writer["config"]["role_name"] == "Worker"
99
+ assert "write_dirs" not in reader["config"]
100
+ assert "allow_network" not in reader["config"]
101
+
102
+ removed_permission_response = client.post(
103
+ f"/api/workflows/{tab_id}/nodes",
104
+ json={
105
+ "role_name": "Worker",
106
+ "write_dirs": ["/tmp"],
107
+ },
108
+ )
109
+ assert removed_permission_response.status_code == 422
110
+
111
+ edge_response = client.post(
112
+ f"/api/workflows/{tab_id}/edges",
113
+ json={
114
+ "from_node_id": reader["id"],
115
+ "from_port_key": "out",
116
+ "to_node_id": writer["id"],
117
+ "to_port_key": "in",
118
+ },
119
+ )
120
+
121
+ assert edge_response.status_code == 200
122
+ edge = edge_response.json()
123
+ assert edge["tab_id"] == tab_id
124
+ assert edge["from_node_id"] == reader["id"]
125
+ assert edge["from_port_key"] == "out"
126
+ assert edge["to_node_id"] == writer["id"]
127
+ assert edge["to_port_key"] == "in"
128
+
129
+ tab_detail_response = client.get(f"/api/workflows/{tab_id}")
130
+ assert tab_detail_response.status_code == 200
131
+ tab_detail = tab_detail_response.json()
132
+ assert tab_detail["workflow"]["id"] == tab_id
133
+ assert "goal" not in tab_detail["workflow"]
134
+ assert tab_detail["workflow"]["allow_network"] is True
135
+ assert tab_detail["workflow"]["write_dirs"] == ["/tmp"]
136
+ assert tab_detail["workflow"]["node_count"] == 2
137
+ assert tab_detail["workflow"]["edge_count"] == 1
138
+ assert {node["name"] for node in tab_detail["nodes"]} == {"Reader", "Writer"}
139
+ assert tab_detail["edges"] == [edge]
140
+ assert len(tab_detail["workflow"]["definition"]["nodes"]) == 2
141
+ assert tab_detail["workflow"]["definition"]["edges"] == [edge]
142
+
143
+ nodes_response = client.get("/api/nodes")
144
+ assert nodes_response.status_code == 200
145
+ nodes = nodes_response.json()["nodes"]
146
+ reader_node = next(node for node in nodes if node["id"] == reader["id"])
147
+ writer_node = next(node for node in nodes if node["id"] == writer["id"])
148
+ assert reader_node["workflow_id"] == tab_id
149
+ assert writer_node["workflow_id"] == tab_id
150
+ assert reader_node["connections"] == [writer["id"]]
151
+ assert writer_node["connections"] == [reader["id"]]
152
+
153
+
154
+ def test_delete_tab_cleans_up_nodes_and_edges(client: TestClient):
155
+ create_tab_response = client.post(
156
+ "/api/workflows",
157
+ json={"title": "Disposable"},
158
+ )
159
+
160
+ assert create_tab_response.status_code == 200
161
+ created_tab = create_tab_response.json()
162
+ tab_id = created_tab["id"]
163
+ leader_id = created_tab["leader_id"]
164
+
165
+ left = _create_agent_node(client, tab_id=tab_id, name="Left")
166
+ right = _create_agent_node(client, tab_id=tab_id, name="Right")
167
+
168
+ edge_response = client.post(
169
+ f"/api/workflows/{tab_id}/edges",
170
+ json={"from_node_id": left["id"], "to_node_id": right["id"]},
171
+ )
172
+ assert edge_response.status_code == 200
173
+ edge_id = edge_response.json()["id"]
174
+
175
+ delete_response = client.delete(f"/api/workflows/{tab_id}")
176
+
177
+ assert delete_response.status_code == 200
178
+ assert delete_response.json()["id"] == tab_id
179
+ assert set(delete_response.json()["removed_node_ids"]) == {
180
+ leader_id,
181
+ left["id"],
182
+ right["id"],
183
+ }
184
+ assert delete_response.json()["removed_edge_ids"] == [edge_id]
185
+
186
+ tab_detail_response = client.get(f"/api/workflows/{tab_id}")
187
+ assert tab_detail_response.status_code == 404
188
+
189
+ nodes_response = client.get("/api/nodes")
190
+ assert nodes_response.status_code == 200
191
+ node_ids = {node["id"] for node in nodes_response.json()["nodes"]}
192
+ assert left["id"] not in node_ids
193
+ assert right["id"] not in node_ids
194
+
195
+
196
+ def test_delete_tab_edge_requires_exact_direction_and_removes_only_target_edge(
197
+ client: TestClient,
198
+ ):
199
+ tab = client.post(
200
+ "/api/workflows",
201
+ json={"title": "Edge Delete"},
202
+ ).json()
203
+ tab_id = tab["id"]
204
+
205
+ left = _create_agent_node(client, tab_id=tab_id, name="Left")
206
+ middle = _create_agent_node(client, tab_id=tab_id, name="Middle")
207
+ right = _create_agent_node(client, tab_id=tab_id, name="Right")
208
+
209
+ left_to_middle = client.post(
210
+ f"/api/workflows/{tab_id}/edges",
211
+ json={"from_node_id": left["id"], "to_node_id": middle["id"]},
212
+ )
213
+ middle_to_right = client.post(
214
+ f"/api/workflows/{tab_id}/edges",
215
+ json={"from_node_id": middle["id"], "to_node_id": right["id"]},
216
+ )
217
+
218
+ assert left_to_middle.status_code == 200
219
+ assert middle_to_right.status_code == 200
220
+
221
+ reverse_delete_response = client.delete(
222
+ f"/api/workflows/{tab_id}/edges",
223
+ params={
224
+ "from_node_id": middle["id"],
225
+ "to_node_id": left["id"],
226
+ },
227
+ )
228
+ assert reverse_delete_response.status_code == 404
229
+ assert reverse_delete_response.json()["detail"] == "Edge not found"
230
+
231
+ delete_response = client.delete(
232
+ f"/api/workflows/{tab_id}/edges",
233
+ params={
234
+ "from_node_id": left["id"],
235
+ "to_node_id": middle["id"],
236
+ },
237
+ )
238
+
239
+ assert delete_response.status_code == 200
240
+ assert delete_response.json()["from_node_id"] == left["id"]
241
+ assert delete_response.json()["to_node_id"] == middle["id"]
242
+
243
+ detail = client.get(f"/api/workflows/{tab_id}").json()
244
+ remaining_edges = {
245
+ (edge["from_node_id"], edge["to_node_id"]) for edge in detail["edges"]
246
+ }
247
+ assert remaining_edges == {(middle["id"], right["id"])}
248
+
249
+
250
+ def test_delete_tab_node_removes_node_and_all_incident_edges(client: TestClient):
251
+ tab = client.post(
252
+ "/api/workflows",
253
+ json={"title": "Node Delete"},
254
+ ).json()
255
+ tab_id = tab["id"]
256
+
257
+ left = _create_agent_node(client, tab_id=tab_id, name="Left")
258
+ middle = _create_agent_node(client, tab_id=tab_id, name="Middle")
259
+ right = _create_agent_node(client, tab_id=tab_id, name="Right")
260
+
261
+ assert (
262
+ client.post(
263
+ f"/api/workflows/{tab_id}/edges",
264
+ json={"from_node_id": left["id"], "to_node_id": middle["id"]},
265
+ ).status_code
266
+ == 200
267
+ )
268
+ assert (
269
+ client.post(
270
+ f"/api/workflows/{tab_id}/edges",
271
+ json={"from_node_id": middle["id"], "to_node_id": right["id"]},
272
+ ).status_code
273
+ == 200
274
+ )
275
+
276
+ delete_response = client.delete(f"/api/workflows/{tab_id}/nodes/{middle['id']}")
277
+
278
+ assert delete_response.status_code == 200
279
+ assert delete_response.json()["id"] == middle["id"]
280
+
281
+ detail = client.get(f"/api/workflows/{tab_id}").json()
282
+ remaining_nodes = {node["id"] for node in detail["nodes"]}
283
+ assert middle["id"] not in remaining_nodes
284
+ assert left["id"] in remaining_nodes
285
+ assert right["id"] in remaining_nodes
286
+ assert detail["edges"] == []
287
+
288
+
289
+ def test_tab_edge_creation_enforces_directed_ports_and_single_input(
290
+ client: TestClient,
291
+ ):
292
+ tab = client.post(
293
+ "/api/workflows",
294
+ json={"title": "Edge Validation"},
295
+ ).json()
296
+ tab_id = tab["id"]
297
+ worker = _create_agent_node(client, tab_id=tab_id, name="Worker")
298
+ reviewer = _create_agent_node(client, tab_id=tab_id, name="Reviewer")
299
+ observer = _create_agent_node(client, tab_id=tab_id, name="Observer")
300
+
301
+ self_loop_response = client.post(
302
+ f"/api/workflows/{tab_id}/edges",
303
+ json={"from_node_id": worker["id"], "to_node_id": worker["id"]},
304
+ )
305
+ assert self_loop_response.status_code == 400
306
+ assert self_loop_response.json()["detail"] == "Self-loop edges are not allowed"
307
+
308
+ first_edge_response = client.post(
309
+ f"/api/workflows/{tab_id}/edges",
310
+ json={"from_node_id": worker["id"], "to_node_id": reviewer["id"]},
311
+ )
312
+ assert first_edge_response.status_code == 200
313
+
314
+ duplicate_edge_response = client.post(
315
+ f"/api/workflows/{tab_id}/edges",
316
+ json={"from_node_id": worker["id"], "to_node_id": reviewer["id"]},
317
+ )
318
+ assert duplicate_edge_response.status_code == 400
319
+ assert duplicate_edge_response.json()["detail"] == "Duplicate edges are not allowed"
320
+
321
+ reverse_edge_response = client.post(
322
+ f"/api/workflows/{tab_id}/edges",
323
+ json={"from_node_id": reviewer["id"], "to_node_id": worker["id"]},
324
+ )
325
+ assert reverse_edge_response.status_code == 200
326
+
327
+ conflicting_input_response = client.post(
328
+ f"/api/workflows/{tab_id}/edges",
329
+ json={"from_node_id": observer["id"], "to_node_id": reviewer["id"]},
330
+ )
331
+ assert conflicting_input_response.status_code == 400
332
+ assert (
333
+ conflicting_input_response.json()["detail"]
334
+ == "Input port 'in' already has an incoming edge"
335
+ )
336
+
337
+
338
+ def test_duplicate_tab_copies_definition_and_runtime_agents(client: TestClient):
339
+ source_tab = client.post(
340
+ "/api/workflows",
341
+ json={
342
+ "title": "Original Workflow",
343
+ "allow_network": True,
344
+ "write_dirs": ["/tmp"],
345
+ },
346
+ ).json()
347
+ source_tab_id = source_tab["id"]
348
+
349
+ reviewer = _create_agent_node(client, tab_id=source_tab_id, name="Reviewer")
350
+ formatter = _create_graph_node(
351
+ client,
352
+ tab_id=source_tab_id,
353
+ node_type="code",
354
+ name="Formatter",
355
+ config={"language": "python"},
356
+ )
357
+ source_detail = client.get(f"/api/workflows/{source_tab_id}").json()
358
+ source_definition = deepcopy(source_detail["workflow"]["definition"])
359
+ source_definition["view"] = {
360
+ "positions": {
361
+ reviewer["id"]: {"x": 20, "y": 40},
362
+ formatter["id"]: {"x": 180, "y": 40},
363
+ }
364
+ }
365
+ source_definition["edges"] = [
366
+ {
367
+ "id": "edge-review",
368
+ "from_node_id": reviewer["id"],
369
+ "from_port_key": "out",
370
+ "to_node_id": formatter["id"],
371
+ "to_port_key": "in",
372
+ }
373
+ ]
374
+ update_response = client.put(
375
+ f"/api/workflows/{source_tab_id}/definition",
376
+ json={"definition": source_definition},
377
+ )
378
+ assert update_response.status_code == 200
379
+
380
+ duplicate_response = client.post(f"/api/workflows/{source_tab_id}/duplicate")
381
+
382
+ assert duplicate_response.status_code == 200
383
+ duplicated_tab = duplicate_response.json()
384
+ assert duplicated_tab["title"] == "Original Workflow Copy"
385
+ assert "goal" not in duplicated_tab
386
+ assert duplicated_tab["node_count"] == 2
387
+ assert duplicated_tab["edge_count"] == 1
388
+ assert duplicated_tab["id"] != source_tab_id
389
+ assert duplicated_tab["leader_id"] != source_tab["leader_id"]
390
+ assert duplicated_tab["activation_state"] == "inactive"
391
+ assert duplicated_tab["allow_network"] is True
392
+ assert duplicated_tab["write_dirs"] == ["/tmp"]
393
+
394
+ duplicated_detail = client.get(f"/api/workflows/{duplicated_tab['id']}").json()
395
+ assert "goal" not in duplicated_detail["workflow"]
396
+ assert duplicated_detail["workflow"]["allow_network"] is True
397
+ assert duplicated_detail["workflow"]["write_dirs"] == ["/tmp"]
398
+ assert {node["name"] for node in duplicated_detail["nodes"]} == {
399
+ "Reviewer",
400
+ "Formatter",
401
+ }
402
+ duplicated_node_ids = {node["id"] for node in duplicated_detail["nodes"]}
403
+ source_node_ids = {node["id"] for node in source_detail["nodes"]}
404
+ assert duplicated_node_ids.isdisjoint(source_node_ids)
405
+ assert duplicated_detail["edges"][0]["from_node_id"] in duplicated_node_ids
406
+ assert duplicated_detail["edges"][0]["to_node_id"] in duplicated_node_ids
407
+ assert duplicated_detail["workflow"]["definition"]["view"]["positions"]
408
+
409
+
410
+ def test_update_tab_definition_updates_metadata_and_positions(client: TestClient):
411
+ tab = client.post(
412
+ "/api/workflows",
413
+ json={"title": "JSON Editor"},
414
+ ).json()
415
+ tab_id = tab["id"]
416
+
417
+ agent_node = _create_agent_node(client, tab_id=tab_id, name="Draft Reviewer")
418
+ code_node = _create_graph_node(
419
+ client,
420
+ tab_id=tab_id,
421
+ node_type="code",
422
+ name="Formatter",
423
+ )
424
+
425
+ current = client.get(f"/api/workflows/{tab_id}").json()["workflow"]["definition"]
426
+ definition = deepcopy(current)
427
+ for node in definition["nodes"]:
428
+ if node["id"] == agent_node["id"]:
429
+ node["config"]["name"] = "Final Reviewer"
430
+ if node["id"] == code_node["id"]:
431
+ node["config"]["name"] = "Formatter"
432
+ node["config"]["language"] = "python"
433
+ definition["view"] = {
434
+ "positions": {
435
+ agent_node["id"]: {"x": 60, "y": 80},
436
+ code_node["id"]: {"x": 260, "y": 80},
437
+ }
438
+ }
439
+ definition["edges"] = [
440
+ {
441
+ "id": "edge-control",
442
+ "from_node_id": agent_node["id"],
443
+ "from_port_key": "out",
444
+ "to_node_id": code_node["id"],
445
+ "to_port_key": "in",
446
+ }
447
+ ]
448
+
449
+ update_response = client.put(
450
+ f"/api/workflows/{tab_id}/definition",
451
+ json={"definition": definition},
452
+ )
453
+
454
+ assert update_response.status_code == 200
455
+ updated = update_response.json()
456
+ assert updated["definition"]["view"]["positions"][agent_node["id"]] == {
457
+ "x": 60.0,
458
+ "y": 80.0,
459
+ }
460
+ assert "kind" not in updated["definition"]["edges"][0]
461
+
462
+ detail = client.get(f"/api/workflows/{tab_id}").json()
463
+ reviewer_detail = next(
464
+ node for node in detail["nodes"] if node["id"] == agent_node["id"]
465
+ )
466
+ formatter_detail = next(
467
+ node for node in detail["nodes"] if node["id"] == code_node["id"]
468
+ )
469
+ assert reviewer_detail["name"] == "Final Reviewer"
470
+ assert reviewer_detail["position"] == {"x": 60.0, "y": 80.0}
471
+ assert formatter_detail["config"]["language"] == "python"
472
+ assert formatter_detail["position"] == {"x": 260.0, "y": 80.0}
473
+
474
+ runtime_node = client.get(f"/api/nodes/{agent_node['id']}")
475
+ assert runtime_node.status_code == 200
476
+ assert runtime_node.json()["name"] == "Final Reviewer"
477
+
478
+
479
+ def test_update_tab_definition_rejects_agent_set_changes(client: TestClient):
480
+ tab = client.post(
481
+ "/api/workflows",
482
+ json={"title": "Guard Rails"},
483
+ ).json()
484
+ tab_id = tab["id"]
485
+
486
+ agent_node = _create_agent_node(client, tab_id=tab_id, name="Existing Worker")
487
+ definition = deepcopy(
488
+ client.get(f"/api/workflows/{tab_id}").json()["workflow"]["definition"]
489
+ )
490
+ definition["nodes"].append(
491
+ {
492
+ "id": "new-agent",
493
+ "type": "agent",
494
+ "config": {"role_name": "Worker", "name": "Injected Worker"},
495
+ "inputs": [
496
+ {
497
+ "key": "in",
498
+ "direction": "in",
499
+ "type": "parts",
500
+ "required": False,
501
+ "multiple": False,
502
+ }
503
+ ],
504
+ "outputs": [
505
+ {
506
+ "key": "out",
507
+ "direction": "out",
508
+ "type": "parts",
509
+ "required": False,
510
+ "multiple": True,
511
+ }
512
+ ],
513
+ }
514
+ )
515
+
516
+ response = client.put(
517
+ f"/api/workflows/{tab_id}/definition",
518
+ json={"definition": definition},
519
+ )
520
+
521
+ assert response.status_code == 400
522
+ assert response.json()["detail"] == (
523
+ "Agent nodes must be created or deleted through workflow node APIs"
524
+ )
525
+ untouched = client.get(f"/api/nodes/{agent_node['id']}")
526
+ assert untouched.status_code == 200
527
+
528
+
529
+ def test_activate_empty_workflow_fails_with_validation_errors(client: TestClient):
530
+ tab = client.post("/api/workflows", json={"title": "Empty"}).json()
531
+
532
+ response = client.post(f"/api/workflows/{tab['id']}/activate")
533
+
534
+ assert response.status_code == 400
535
+ errors = response.json()["detail"]["errors"]
536
+ assert any(
537
+ error["message"] == "Add at least one node before activating this workflow"
538
+ for error in errors
539
+ )
540
+ detail = client.get(f"/api/workflows/{tab['id']}").json()
541
+ assert detail["workflow"]["activation_state"] == "inactive"
542
+
543
+
544
+ def test_activate_agent_only_workflow_succeeds_without_trigger(client: TestClient):
545
+ tab = client.post("/api/workflows", json={"title": "Collaborative"}).json()
546
+ worker = _create_agent_node(client, tab_id=tab["id"], name="Worker")
547
+
548
+ response = client.post(f"/api/workflows/{tab['id']}/activate")
549
+
550
+ assert response.status_code == 200
551
+ assert response.json()["activation_state"] == "active"
552
+ detail = client.get(f"/api/workflows/{tab['id']}").json()
553
+ assert detail["workflow"]["activation_state"] == "active"
554
+ assert detail["nodes"][0]["id"] == worker["id"]
555
+
556
+
557
+ def test_activate_legacy_agent_only_workflow_succeeds_without_trigger(
558
+ client: TestClient,
559
+ ):
560
+ tab = client.post("/api/workflows", json={"title": "Legacy Collaborative"}).json()
561
+ worker = _create_agent_node(client, tab_id=tab["id"], name="Worker")
562
+ definition = deepcopy(
563
+ client.get(f"/api/workflows/{tab['id']}").json()["workflow"]["definition"]
564
+ )
565
+ definition["nodes"][0]["inputs"][0]["required"] = True
566
+ update_response = client.put(
567
+ f"/api/workflows/{tab['id']}/definition",
568
+ json={"definition": definition},
569
+ )
570
+ assert update_response.status_code == 200
571
+
572
+ response = client.post(f"/api/workflows/{tab['id']}/activate")
573
+
574
+ assert response.status_code == 200
575
+ assert response.json()["activation_state"] == "active"
576
+ detail = client.get(f"/api/workflows/{tab['id']}").json()
577
+ assert detail["workflow"]["activation_state"] == "active"
578
+ assert detail["nodes"][0]["id"] == worker["id"]
579
+
580
+
581
+ def test_activate_valid_manual_trigger_graph_succeeds(client: TestClient):
582
+ tab = client.post("/api/workflows", json={"title": "Manual"}).json()
583
+ trigger = _create_graph_node(
584
+ client,
585
+ tab_id=tab["id"],
586
+ node_type="trigger",
587
+ name="Manual start",
588
+ config={"kind": "manual", "output_type": "string", "message": "Run"},
589
+ )
590
+
591
+ response = client.post(f"/api/workflows/{tab['id']}/activate")
592
+
593
+ assert response.status_code == 200
594
+ assert response.json()["activation_state"] == "active"
595
+ detail = client.get(f"/api/workflows/{tab['id']}").json()
596
+ assert detail["workflow"]["activation_state"] == "active"
597
+ assert detail["nodes"][0]["id"] == trigger["id"]
598
+
599
+
600
+ def test_active_workflow_locks_semantic_edits_but_allows_view_updates(
601
+ client: TestClient,
602
+ ):
603
+ tab = client.post("/api/workflows", json={"title": "Locked"}).json()
604
+ trigger = _create_graph_node(
605
+ client,
606
+ tab_id=tab["id"],
607
+ node_type="trigger",
608
+ name="Manual start",
609
+ config={"kind": "manual", "output_type": "string", "message": "Run"},
610
+ )
611
+ assert client.post(f"/api/workflows/{tab['id']}/activate").status_code == 200
612
+
613
+ create_response = client.post(
614
+ f"/api/workflows/{tab['id']}/nodes",
615
+ json={"node_type": "trigger", "config": {"kind": "manual"}},
616
+ )
617
+ assert create_response.status_code == 400
618
+ assert "active" in create_response.json()["detail"]
619
+
620
+ definition = deepcopy(
621
+ client.get(f"/api/workflows/{tab['id']}").json()["workflow"]["definition"]
622
+ )
623
+ definition["nodes"][0]["config"]["message"] = "Changed"
624
+ semantic_response = client.put(
625
+ f"/api/workflows/{tab['id']}/definition",
626
+ json={"definition": definition},
627
+ )
628
+ assert semantic_response.status_code == 400
629
+ assert "active" in semantic_response.json()["detail"]
630
+
631
+ view_definition = deepcopy(
632
+ client.get(f"/api/workflows/{tab['id']}").json()["workflow"]["definition"]
633
+ )
634
+ view_definition["view"] = {"positions": {trigger["id"]: {"x": 12, "y": 24}}}
635
+ view_response = client.put(
636
+ f"/api/workflows/{tab['id']}/definition",
637
+ json={"definition": view_definition},
638
+ )
639
+ assert view_response.status_code == 200
640
+ assert view_response.json()["activation_state"] == "active"
641
+ assert view_response.json()["definition"]["view"]["positions"][trigger["id"]] == {
642
+ "x": 12.0,
643
+ "y": 24.0,
644
+ }
645
+
646
+
647
+ def test_deactivate_interrupts_running_workflow_nodes(
648
+ monkeypatch,
649
+ client: TestClient,
650
+ ):
651
+ from flowent.models import AgentState
652
+ from flowent.registry import registry
653
+
654
+ tab = client.post("/api/workflows", json={"title": "Deactivate"}).json()
655
+ trigger = _create_graph_node(
656
+ client,
657
+ tab_id=tab["id"],
658
+ node_type="trigger",
659
+ name="Manual start",
660
+ config={
661
+ "kind": "manual",
662
+ "output_type": "parts",
663
+ "message": [{"type": "text", "text": "Run"}],
664
+ },
665
+ )
666
+ worker = _create_agent_node(client, tab_id=tab["id"], name="Worker")
667
+ assert (
668
+ client.post(
669
+ f"/api/workflows/{tab['id']}/edges",
670
+ json={"from_node_id": trigger["id"], "to_node_id": worker["id"]},
671
+ ).status_code
672
+ == 200
673
+ )
674
+ assert client.post(f"/api/workflows/{tab['id']}/activate").status_code == 200
675
+ live_worker = registry.get(worker["id"])
676
+ assert live_worker is not None
677
+ live_worker.set_state(AgentState.RUNNING, "test")
678
+ interrupted: list[str] = []
679
+
680
+ def fake_request_interrupt() -> bool:
681
+ interrupted.append(live_worker.uuid)
682
+ live_worker.set_state(AgentState.IDLE, "interrupted")
683
+ return True
684
+
685
+ monkeypatch.setattr(live_worker, "request_interrupt", fake_request_interrupt)
686
+
687
+ response = client.post(f"/api/workflows/{tab['id']}/deactivate")
688
+
689
+ assert response.status_code == 200
690
+ assert response.json()["activation_state"] == "inactive"
691
+ assert interrupted == [worker["id"]]
692
+ assert live_worker.state == AgentState.IDLE
693
+
694
+
695
+ def test_llm_node_and_typed_port_validation(client: TestClient):
696
+ from flowent.models import GraphEdge
697
+ from flowent.workspace_store import workspace_store
698
+
699
+ provider = client.post(
700
+ "/api/providers",
701
+ json={
702
+ "name": "Primary",
703
+ "type": "openai_compatible",
704
+ "base_url": "https://api.example.com",
705
+ "models": [{"model": "gpt-5", "structured_output": True}],
706
+ },
707
+ ).json()
708
+ tab = client.post("/api/workflows", json={"title": "Typed"}).json()
709
+ trigger = _create_graph_node(
710
+ client,
711
+ tab_id=tab["id"],
712
+ node_type="trigger",
713
+ name="Text trigger",
714
+ config={"kind": "manual", "output_type": "string", "message": "Run"},
715
+ )
716
+ llm = _create_graph_node(
717
+ client,
718
+ tab_id=tab["id"],
719
+ node_type="llm",
720
+ name="JSON reader",
721
+ config={
722
+ "model": {"provider_id": provider["id"], "model": "gpt-5"},
723
+ "system_prompt": "Read input.",
724
+ "temperature": 0,
725
+ "max_output_tokens": 100,
726
+ "stop_sequences": [],
727
+ "response_format": {"kind": "text"},
728
+ "input_type": "json",
729
+ "output_type": "string",
730
+ },
731
+ )
732
+ stored_tab = workspace_store.get_tab(tab["id"])
733
+ assert stored_tab is not None
734
+ stored_tab.definition.edges.append(
735
+ GraphEdge(
736
+ id="invalid-edge",
737
+ tab_id=tab["id"],
738
+ from_node_id=trigger["id"],
739
+ from_port_key="out",
740
+ to_node_id=llm["id"],
741
+ to_port_key="in",
742
+ )
743
+ )
744
+ workspace_store.upsert_tab(stored_tab)
745
+
746
+ response = client.post(f"/api/workflows/{tab['id']}/activate")
747
+
748
+ assert response.status_code == 400
749
+ messages = [error["message"] for error in response.json()["detail"]["errors"]]
750
+ assert any("port type mismatch" in message for message in messages)
751
+
752
+
753
+ def test_structured_output_false_blocks_json_schema_llm(client: TestClient):
754
+ provider = client.post(
755
+ "/api/providers",
756
+ json={
757
+ "name": "Primary",
758
+ "type": "openai_compatible",
759
+ "base_url": "https://api.example.com",
760
+ "models": [{"model": "gpt-5", "structured_output": False}],
761
+ },
762
+ ).json()
763
+ tab = client.post("/api/workflows", json={"title": "Structured"}).json()
764
+ trigger = _create_graph_node(
765
+ client,
766
+ tab_id=tab["id"],
767
+ node_type="trigger",
768
+ name="JSON trigger",
769
+ config={"kind": "manual", "output_type": "json", "message": {"task": "Run"}},
770
+ )
771
+ llm = _create_graph_node(
772
+ client,
773
+ tab_id=tab["id"],
774
+ node_type="llm",
775
+ name="JSON writer",
776
+ config={
777
+ "model": {"provider_id": provider["id"], "model": "gpt-5"},
778
+ "system_prompt": "Return JSON.",
779
+ "temperature": 0,
780
+ "max_output_tokens": 100,
781
+ "stop_sequences": [],
782
+ "response_format": {
783
+ "kind": "json_schema",
784
+ "schema": {"type": "object"},
785
+ },
786
+ "input_type": "json",
787
+ "output_type": "json",
788
+ },
789
+ )
790
+ assert (
791
+ client.post(
792
+ f"/api/workflows/{tab['id']}/edges",
793
+ json={"from_node_id": trigger["id"], "to_node_id": llm["id"]},
794
+ ).status_code
795
+ == 200
796
+ )
797
+
798
+ response = client.post(f"/api/workflows/{tab['id']}/activate")
799
+
800
+ assert response.status_code == 400
801
+ messages = [error["message"] for error in response.json()["detail"]["errors"]]
802
+ assert "llm model does not support structured_output" in messages