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,595 @@
1
+ import json
2
+
3
+ import pytest
4
+
5
+ from flowent.agent import Agent
6
+ from flowent.models import AgentState, GraphNodeRecord, NodeConfig, NodeType, Tab
7
+ from flowent.registry import registry
8
+ from flowent.settings import AssistantSettings, Settings
9
+ from flowent.tools.set_permissions import SetPermissionsTool
10
+ from flowent.workspace_store import workspace_store
11
+
12
+
13
+ @pytest.fixture(autouse=True)
14
+ def reset_runtime_state(monkeypatch, tmp_path):
15
+ import flowent.settings as settings_module
16
+
17
+ settings_file = tmp_path / "settings.json"
18
+ settings_file.write_text("{}", encoding="utf-8")
19
+ monkeypatch.setattr(settings_module, "_SETTINGS_FILE", settings_file)
20
+ monkeypatch.setattr(settings_module, "_cached_settings", None)
21
+ registry.reset()
22
+ workspace_store.reset_cache()
23
+ yield
24
+ registry.reset()
25
+ workspace_store.reset_cache()
26
+ monkeypatch.setattr(settings_module, "_cached_settings", None)
27
+
28
+
29
+ def _make_record(
30
+ *,
31
+ node_id: str,
32
+ tab_id: str,
33
+ role_name: str,
34
+ name: str,
35
+ write_dirs: list[str],
36
+ allow_network: bool,
37
+ tools: list[str] | None = None,
38
+ ) -> GraphNodeRecord:
39
+ return GraphNodeRecord(
40
+ id=node_id,
41
+ config=NodeConfig(
42
+ node_type=NodeType.AGENT,
43
+ role_name=role_name,
44
+ tab_id=tab_id,
45
+ name=name,
46
+ tools=list(tools or []),
47
+ write_dirs=list(write_dirs),
48
+ allow_network=allow_network,
49
+ ),
50
+ state=AgentState.INITIALIZING,
51
+ )
52
+
53
+
54
+ def _register_live_node(record: GraphNodeRecord) -> Agent:
55
+ node = Agent(
56
+ NodeConfig(
57
+ node_type=record.config.node_type,
58
+ role_name=record.config.role_name,
59
+ tab_id=record.config.tab_id,
60
+ name=record.config.name,
61
+ tools=list(record.config.tools),
62
+ write_dirs=list(record.config.write_dirs),
63
+ allow_network=record.config.allow_network,
64
+ ),
65
+ uuid=record.id,
66
+ )
67
+ registry.register(node)
68
+ return node
69
+
70
+
71
+ def _grant_assistant_permissions(
72
+ monkeypatch,
73
+ *,
74
+ write_dirs: list[str],
75
+ allow_network: bool,
76
+ ) -> None:
77
+ monkeypatch.setattr(
78
+ "flowent.settings.get_settings",
79
+ lambda: Settings(
80
+ assistant=AssistantSettings(
81
+ write_dirs=list(write_dirs),
82
+ allow_network=allow_network,
83
+ )
84
+ ),
85
+ )
86
+
87
+
88
+ def test_set_permissions_updates_workflow_boundary_for_all_nodes(monkeypatch, tmp_path):
89
+ root_dir = tmp_path / "root"
90
+ keep_boundary = root_dir / "keep"
91
+ keep_dir = keep_boundary / "child"
92
+ drop_dir = root_dir / "drop"
93
+ keep_dir.mkdir(parents=True)
94
+ drop_dir.mkdir(parents=True)
95
+ _grant_assistant_permissions(
96
+ monkeypatch,
97
+ write_dirs=[str(root_dir)],
98
+ allow_network=True,
99
+ )
100
+
101
+ tab = Tab(id="tab-1", title="Task", leader_id="leader-1")
102
+ tab.allow_network = True
103
+ tab.write_dirs = [str(root_dir)]
104
+ tab.permissions_initialized = True
105
+ workspace_store.upsert_tab(tab)
106
+ leader = _make_record(
107
+ node_id="leader-1",
108
+ tab_id=tab.id,
109
+ role_name="Conductor",
110
+ name="Leader",
111
+ write_dirs=[str(root_dir)],
112
+ allow_network=True,
113
+ )
114
+ keep_worker = _make_record(
115
+ node_id="worker-keep",
116
+ tab_id=tab.id,
117
+ role_name="Worker",
118
+ name="Keep Worker",
119
+ write_dirs=[str(keep_dir)],
120
+ allow_network=True,
121
+ )
122
+ drop_worker = _make_record(
123
+ node_id="worker-drop",
124
+ tab_id=tab.id,
125
+ role_name="Worker",
126
+ name="Drop Worker",
127
+ write_dirs=[str(drop_dir)],
128
+ allow_network=True,
129
+ )
130
+ workspace_store.upsert_node_record(leader)
131
+ workspace_store.upsert_node_record(keep_worker)
132
+ workspace_store.upsert_node_record(drop_worker)
133
+ leader_live = _register_live_node(leader)
134
+ keep_live = _register_live_node(keep_worker)
135
+ drop_live = _register_live_node(drop_worker)
136
+ assistant = Agent(
137
+ NodeConfig(
138
+ node_type=NodeType.ASSISTANT,
139
+ tools=["set_permissions"],
140
+ write_dirs=[str(root_dir)],
141
+ allow_network=True,
142
+ ),
143
+ uuid="assistant",
144
+ )
145
+
146
+ result = json.loads(
147
+ SetPermissionsTool().execute(
148
+ assistant,
149
+ {
150
+ "workflow_id": tab.id,
151
+ "allow_network": False,
152
+ "write_dirs": [str(keep_boundary)],
153
+ },
154
+ )
155
+ )
156
+
157
+ assert result == {
158
+ "tab_id": tab.id,
159
+ "leader_id": "leader-1",
160
+ "allow_network": False,
161
+ "write_dirs": [str(keep_boundary)],
162
+ "updated_node_ids": ["leader-1", "worker-keep", "worker-drop"],
163
+ }
164
+ updated_tab = workspace_store.get_tab(tab.id)
165
+ assert updated_tab is not None
166
+ assert updated_tab.allow_network is False
167
+ assert updated_tab.write_dirs == [str(keep_boundary)]
168
+ assert leader_live.config.allow_network is True
169
+ assert leader_live.config.write_dirs == [str(root_dir)]
170
+ assert keep_live.config.allow_network is True
171
+ assert keep_live.config.write_dirs == [str(keep_dir)]
172
+ assert drop_live.config.allow_network is True
173
+ assert drop_live.config.write_dirs == [str(drop_dir)]
174
+
175
+
176
+ def test_set_permissions_keeps_omitted_fields_unchanged(monkeypatch, tmp_path):
177
+ root_dir = tmp_path / "root"
178
+ narrowed_dir = root_dir / "narrowed"
179
+ root_dir.mkdir()
180
+ narrowed_dir.mkdir()
181
+ _grant_assistant_permissions(
182
+ monkeypatch,
183
+ write_dirs=[str(root_dir)],
184
+ allow_network=False,
185
+ )
186
+
187
+ tab = Tab(id="tab-1", title="Task", leader_id="leader-1")
188
+ tab.allow_network = True
189
+ tab.write_dirs = [str(root_dir)]
190
+ tab.permissions_initialized = True
191
+ workspace_store.upsert_tab(tab)
192
+ leader = _make_record(
193
+ node_id="leader-1",
194
+ tab_id=tab.id,
195
+ role_name="Conductor",
196
+ name="Leader",
197
+ write_dirs=[str(root_dir)],
198
+ allow_network=True,
199
+ )
200
+ workspace_store.upsert_node_record(leader)
201
+ leader_live = _register_live_node(leader)
202
+ assistant = Agent(
203
+ NodeConfig(
204
+ node_type=NodeType.ASSISTANT,
205
+ tools=["set_permissions"],
206
+ write_dirs=[str(root_dir)],
207
+ allow_network=False,
208
+ ),
209
+ uuid="assistant",
210
+ )
211
+
212
+ result = json.loads(
213
+ SetPermissionsTool().execute(
214
+ assistant,
215
+ {
216
+ "workflow_id": tab.id,
217
+ "write_dirs": [str(narrowed_dir)],
218
+ },
219
+ )
220
+ )
221
+
222
+ assert result["allow_network"] is True
223
+ assert result["write_dirs"] == [str(narrowed_dir)]
224
+ updated_tab = workspace_store.get_tab(tab.id)
225
+ assert updated_tab is not None
226
+ assert updated_tab.allow_network is True
227
+ assert updated_tab.write_dirs == [str(narrowed_dir)]
228
+ assert leader_live.config.allow_network is True
229
+ assert leader_live.config.write_dirs == [str(root_dir)]
230
+
231
+
232
+ def test_set_permissions_broadens_workflow_boundary_without_mutating_workers(
233
+ monkeypatch,
234
+ tmp_path,
235
+ ):
236
+ root_dir = tmp_path / "root"
237
+ current_boundary = root_dir / "current"
238
+ worker_dir = current_boundary / "child"
239
+ worker_dir.mkdir(parents=True)
240
+ _grant_assistant_permissions(
241
+ monkeypatch,
242
+ write_dirs=[str(root_dir)],
243
+ allow_network=True,
244
+ )
245
+
246
+ tab = Tab(id="tab-1", title="Task", leader_id="leader-1")
247
+ tab.allow_network = False
248
+ tab.write_dirs = [str(current_boundary)]
249
+ tab.permissions_initialized = True
250
+ workspace_store.upsert_tab(tab)
251
+ leader = _make_record(
252
+ node_id="leader-1",
253
+ tab_id=tab.id,
254
+ role_name="Conductor",
255
+ name="Leader",
256
+ write_dirs=[str(current_boundary)],
257
+ allow_network=False,
258
+ )
259
+ worker = _make_record(
260
+ node_id="worker-1",
261
+ tab_id=tab.id,
262
+ role_name="Worker",
263
+ name="Worker",
264
+ write_dirs=[str(worker_dir)],
265
+ allow_network=False,
266
+ )
267
+ workspace_store.upsert_node_record(leader)
268
+ workspace_store.upsert_node_record(worker)
269
+ worker_live = _register_live_node(worker)
270
+ assistant = Agent(
271
+ NodeConfig(
272
+ node_type=NodeType.ASSISTANT,
273
+ tools=["set_permissions"],
274
+ write_dirs=[str(root_dir)],
275
+ allow_network=True,
276
+ ),
277
+ uuid="assistant",
278
+ )
279
+
280
+ result = json.loads(
281
+ SetPermissionsTool().execute(
282
+ assistant,
283
+ {
284
+ "workflow_id": tab.id,
285
+ "allow_network": True,
286
+ "write_dirs": [str(root_dir)],
287
+ },
288
+ )
289
+ )
290
+
291
+ assert result["allow_network"] is True
292
+ assert result["write_dirs"] == [str(root_dir)]
293
+ updated_tab = workspace_store.get_tab(tab.id)
294
+ assert updated_tab is not None
295
+ assert updated_tab.allow_network is True
296
+ assert updated_tab.write_dirs == [str(root_dir)]
297
+ assert worker_live.config.allow_network is False
298
+ assert worker_live.config.write_dirs == [str(worker_dir)]
299
+
300
+
301
+ def test_set_permissions_rejects_allow_network_outside_caller_boundary(
302
+ monkeypatch,
303
+ tmp_path,
304
+ ):
305
+ root_dir = tmp_path / "root"
306
+ root_dir.mkdir()
307
+ _grant_assistant_permissions(
308
+ monkeypatch,
309
+ write_dirs=[str(root_dir)],
310
+ allow_network=False,
311
+ )
312
+
313
+ tab = Tab(id="tab-1", title="Task", leader_id="leader-1")
314
+ tab.allow_network = False
315
+ tab.write_dirs = [str(root_dir)]
316
+ tab.permissions_initialized = True
317
+ workspace_store.upsert_tab(tab)
318
+ workspace_store.upsert_node_record(
319
+ _make_record(
320
+ node_id="leader-1",
321
+ tab_id=tab.id,
322
+ role_name="Conductor",
323
+ name="Leader",
324
+ write_dirs=[str(root_dir)],
325
+ allow_network=False,
326
+ )
327
+ )
328
+ assistant = Agent(
329
+ NodeConfig(
330
+ node_type=NodeType.ASSISTANT,
331
+ tools=["set_permissions"],
332
+ write_dirs=[str(root_dir)],
333
+ allow_network=False,
334
+ ),
335
+ uuid="assistant",
336
+ )
337
+
338
+ result = json.loads(
339
+ SetPermissionsTool().execute(
340
+ assistant,
341
+ {
342
+ "workflow_id": tab.id,
343
+ "allow_network": True,
344
+ },
345
+ )
346
+ )
347
+
348
+ assert result == {
349
+ "error": "allow_network boundary exceeded: caller disallows network access"
350
+ }
351
+
352
+
353
+ def test_set_permissions_rejects_write_dirs_outside_caller_boundary(
354
+ monkeypatch,
355
+ tmp_path,
356
+ ):
357
+ caller_dir = tmp_path / "caller"
358
+ other_dir = tmp_path / "other"
359
+ caller_dir.mkdir()
360
+ other_dir.mkdir()
361
+ _grant_assistant_permissions(
362
+ monkeypatch,
363
+ write_dirs=[str(caller_dir)],
364
+ allow_network=True,
365
+ )
366
+
367
+ tab = Tab(id="tab-1", title="Task", leader_id="leader-1")
368
+ tab.allow_network = False
369
+ tab.write_dirs = [str(caller_dir)]
370
+ tab.permissions_initialized = True
371
+ workspace_store.upsert_tab(tab)
372
+ workspace_store.upsert_node_record(
373
+ _make_record(
374
+ node_id="leader-1",
375
+ tab_id=tab.id,
376
+ role_name="Conductor",
377
+ name="Leader",
378
+ write_dirs=[str(caller_dir)],
379
+ allow_network=False,
380
+ )
381
+ )
382
+ assistant = Agent(
383
+ NodeConfig(
384
+ node_type=NodeType.ASSISTANT,
385
+ tools=["set_permissions"],
386
+ write_dirs=[str(caller_dir)],
387
+ allow_network=True,
388
+ ),
389
+ uuid="assistant",
390
+ )
391
+
392
+ result = json.loads(
393
+ SetPermissionsTool().execute(
394
+ assistant,
395
+ {
396
+ "workflow_id": tab.id,
397
+ "write_dirs": [str(other_dir)],
398
+ },
399
+ )
400
+ )
401
+
402
+ assert result == {"error": f"write_dirs boundary exceeded: {other_dir}"}
403
+
404
+
405
+ def test_set_permissions_allows_explicitly_granted_non_assistant_agent_for_own_workflow(
406
+ tmp_path,
407
+ ):
408
+ root_dir = tmp_path / "root"
409
+ narrowed_dir = root_dir / "narrowed"
410
+ root_dir.mkdir()
411
+ narrowed_dir.mkdir()
412
+
413
+ tab = Tab(id="tab-1", title="Task", leader_id="leader-1")
414
+ tab.allow_network = True
415
+ tab.write_dirs = [str(root_dir)]
416
+ tab.permissions_initialized = True
417
+ workspace_store.upsert_tab(tab)
418
+ workspace_store.upsert_node_record(
419
+ _make_record(
420
+ node_id="leader-1",
421
+ tab_id=tab.id,
422
+ role_name="Conductor",
423
+ name="Leader",
424
+ write_dirs=[str(root_dir)],
425
+ allow_network=True,
426
+ tools=["set_permissions"],
427
+ )
428
+ )
429
+ leader = Agent(
430
+ NodeConfig(
431
+ node_type=NodeType.AGENT,
432
+ role_name="Conductor",
433
+ tab_id=tab.id,
434
+ name="Leader",
435
+ tools=["set_permissions"],
436
+ write_dirs=[str(root_dir)],
437
+ allow_network=True,
438
+ ),
439
+ uuid="leader-1",
440
+ )
441
+
442
+ result = json.loads(
443
+ SetPermissionsTool().execute(
444
+ leader,
445
+ {
446
+ "workflow_id": tab.id,
447
+ "write_dirs": [str(narrowed_dir)],
448
+ },
449
+ )
450
+ )
451
+
452
+ assert result["tab_id"] == tab.id
453
+ assert result["write_dirs"] == [str(narrowed_dir)]
454
+ updated_tab = workspace_store.get_tab(tab.id)
455
+ assert updated_tab is not None
456
+ assert updated_tab.write_dirs == [str(narrowed_dir)]
457
+
458
+
459
+ def test_set_permissions_rejects_non_assistant_cross_workflow(tmp_path):
460
+ root_dir = tmp_path / "root"
461
+ root_dir.mkdir()
462
+
463
+ own_tab = Tab(id="tab-1", title="Own", leader_id="leader-1")
464
+ own_tab.allow_network = True
465
+ own_tab.write_dirs = [str(root_dir)]
466
+ own_tab.permissions_initialized = True
467
+ other_tab = Tab(id="tab-2", title="Other", leader_id="leader-2")
468
+ other_tab.allow_network = True
469
+ other_tab.write_dirs = [str(root_dir)]
470
+ other_tab.permissions_initialized = True
471
+ workspace_store.upsert_tab(own_tab)
472
+ workspace_store.upsert_tab(other_tab)
473
+ workspace_store.upsert_node_record(
474
+ _make_record(
475
+ node_id="leader-1",
476
+ tab_id=own_tab.id,
477
+ role_name="Conductor",
478
+ name="Leader",
479
+ write_dirs=[str(root_dir)],
480
+ allow_network=True,
481
+ tools=["set_permissions"],
482
+ )
483
+ )
484
+ workspace_store.upsert_node_record(
485
+ _make_record(
486
+ node_id="leader-2",
487
+ tab_id=other_tab.id,
488
+ role_name="Conductor",
489
+ name="Leader",
490
+ write_dirs=[str(root_dir)],
491
+ allow_network=True,
492
+ )
493
+ )
494
+ leader = Agent(
495
+ NodeConfig(
496
+ node_type=NodeType.AGENT,
497
+ role_name="Conductor",
498
+ tab_id=own_tab.id,
499
+ name="Leader",
500
+ tools=["set_permissions"],
501
+ write_dirs=[str(root_dir)],
502
+ allow_network=True,
503
+ ),
504
+ uuid="leader-1",
505
+ )
506
+
507
+ result = json.loads(
508
+ SetPermissionsTool().execute(
509
+ leader,
510
+ {
511
+ "workflow_id": other_tab.id,
512
+ "write_dirs": [],
513
+ },
514
+ )
515
+ )
516
+
517
+ assert result == {
518
+ "error": "Workflow permissions can only be changed for this workflow"
519
+ }
520
+ updated_other_tab = workspace_store.get_tab(other_tab.id)
521
+ assert updated_other_tab is not None
522
+ assert updated_other_tab.write_dirs == [str(root_dir)]
523
+
524
+
525
+ def test_set_permissions_migrates_legacy_leader_permissions(tmp_path):
526
+ root_dir = tmp_path / "root"
527
+ narrowed_dir = root_dir / "narrowed"
528
+ root_dir.mkdir()
529
+ narrowed_dir.mkdir()
530
+
531
+ tab = Tab(id="tab-1", title="Task", leader_id="leader-1")
532
+ workspace_store.upsert_tab(tab)
533
+ workspace_store.upsert_node_record(
534
+ _make_record(
535
+ node_id="leader-1",
536
+ tab_id=tab.id,
537
+ role_name="Conductor",
538
+ name="Leader",
539
+ write_dirs=[str(root_dir)],
540
+ allow_network=True,
541
+ tools=["set_permissions"],
542
+ )
543
+ )
544
+ leader = Agent(
545
+ NodeConfig(
546
+ node_type=NodeType.AGENT,
547
+ role_name="Conductor",
548
+ tab_id=tab.id,
549
+ name="Leader",
550
+ tools=["set_permissions"],
551
+ write_dirs=[str(root_dir)],
552
+ allow_network=True,
553
+ ),
554
+ uuid="leader-1",
555
+ )
556
+
557
+ result = json.loads(
558
+ SetPermissionsTool().execute(
559
+ leader,
560
+ {
561
+ "workflow_id": tab.id,
562
+ "write_dirs": [str(narrowed_dir)],
563
+ },
564
+ )
565
+ )
566
+
567
+ updated_tab = workspace_store.get_tab(tab.id)
568
+ assert result["allow_network"] is True
569
+ assert result["write_dirs"] == [str(narrowed_dir)]
570
+ assert updated_tab is not None
571
+ assert updated_tab.permissions_initialized is True
572
+ assert updated_tab.allow_network is True
573
+ assert updated_tab.write_dirs == [str(narrowed_dir)]
574
+
575
+
576
+ def test_set_permissions_tool_schema_matches_patch_contract():
577
+ assert SetPermissionsTool.parameters == {
578
+ "type": "object",
579
+ "properties": {
580
+ "workflow_id": {
581
+ "type": "string",
582
+ "description": "ID of the workflow whose permission boundary should be updated",
583
+ },
584
+ "allow_network": {
585
+ "type": "boolean",
586
+ "description": "Optional patched network permission for the workflow boundary",
587
+ },
588
+ "write_dirs": {
589
+ "type": "array",
590
+ "items": {"type": "string"},
591
+ "description": "Optional patched writable directory boundary for the workflow",
592
+ },
593
+ },
594
+ "required": ["workflow_id"],
595
+ }
@@ -0,0 +1,37 @@
1
+ import json
2
+
3
+ from flowent.agent import Agent
4
+ from flowent.events import event_bus
5
+ from flowent.models import EventType, NodeConfig, NodeType
6
+ from flowent.tools.todo import TodoTool
7
+
8
+
9
+ def test_todo_tool_emits_node_todos_changed(monkeypatch):
10
+ events = []
11
+ agent = Agent(NodeConfig(node_type=NodeType.AGENT, tools=["todo"]))
12
+
13
+ monkeypatch.setattr(event_bus, "emit", lambda event: events.append(event))
14
+
15
+ result = json.loads(TodoTool().execute(agent, {"todos": ["step 1"]}))
16
+
17
+ assert result == {"status": "updated"}
18
+ assert len(events) == 1
19
+ assert events[0].type == EventType.NODE_TODOS_CHANGED
20
+ assert events[0].data == {
21
+ "todos": [
22
+ {
23
+ "text": "step 1",
24
+ "type": "TodoItem",
25
+ }
26
+ ]
27
+ }
28
+
29
+
30
+ def test_todo_tool_overwrites_and_clears_existing_items():
31
+ agent = Agent(NodeConfig(node_type=NodeType.AGENT, tools=["todo"]))
32
+
33
+ TodoTool().execute(agent, {"todos": ["step 1", "step 2"]})
34
+ result = json.loads(TodoTool().execute(agent, {"todos": []}))
35
+
36
+ assert result == {"status": "updated"}
37
+ assert agent.get_todos_snapshot() == []