flowent 0.0.6 → 0.0.10

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 (398) hide show
  1. package/README.md +1 -4
  2. package/backend/README.md +1 -4
  3. package/backend/pyproject.toml +2 -8
  4. package/backend/src/flowent/__pycache__/__init__.cpython-313.pyc +0 -0
  5. package/backend/src/flowent/__pycache__/agent.cpython-313.pyc +0 -0
  6. package/backend/src/flowent/__pycache__/cli.cpython-313.pyc +0 -0
  7. package/backend/src/flowent/__pycache__/context.cpython-313.pyc +0 -0
  8. package/backend/src/flowent/__pycache__/llm.cpython-313.pyc +0 -0
  9. package/backend/src/flowent/__pycache__/logging.cpython-313.pyc +0 -0
  10. package/backend/src/flowent/__pycache__/main.cpython-313.pyc +0 -0
  11. package/backend/src/flowent/__pycache__/patch.cpython-313.pyc +0 -0
  12. package/backend/src/flowent/__pycache__/paths.cpython-313.pyc +0 -0
  13. package/backend/src/flowent/__pycache__/sandbox.cpython-313.pyc +0 -0
  14. package/backend/src/flowent/__pycache__/storage.cpython-313.pyc +0 -0
  15. package/backend/src/flowent/__pycache__/tools.cpython-313.pyc +0 -0
  16. package/backend/src/flowent/agent.py +217 -3094
  17. package/backend/src/flowent/cli.py +19 -24
  18. package/backend/src/flowent/context.py +127 -0
  19. package/backend/src/flowent/llm.py +256 -0
  20. package/backend/src/flowent/logging.py +170 -129
  21. package/backend/src/flowent/main.py +321 -70
  22. package/backend/src/flowent/patch.py +182 -0
  23. package/backend/src/flowent/paths.py +11 -0
  24. package/backend/src/flowent/sandbox.py +214 -40
  25. package/backend/src/flowent/static/assets/geist-cyrillic-wght-normal-CHSlOQsW.woff2 +0 -0
  26. package/backend/src/flowent/static/assets/geist-latin-ext-wght-normal-DMtmJ5ZE.woff2 +0 -0
  27. package/backend/src/flowent/static/assets/geist-latin-wght-normal-Dm3htQBi.woff2 +0 -0
  28. package/backend/src/flowent/static/assets/index-C76K95ty.js +81 -0
  29. package/backend/src/flowent/static/assets/index-iUMNKvlU.css +2 -0
  30. package/backend/src/flowent/static/flowent.png +0 -0
  31. package/backend/src/flowent/static/index.html +5 -25
  32. package/backend/src/flowent/storage.py +302 -0
  33. package/backend/src/flowent/tools.py +364 -0
  34. package/backend/tests/__pycache__/test_agent_tools.cpython-313-pytest-9.0.3.pyc +0 -0
  35. package/backend/tests/__pycache__/test_health.cpython-313-pytest-9.0.3.pyc +0 -0
  36. package/backend/tests/__pycache__/test_llm_providers.cpython-313-pytest-9.0.3.pyc +0 -0
  37. package/backend/tests/__pycache__/test_logging.cpython-313-pytest-9.0.3.pyc +0 -0
  38. package/backend/tests/__pycache__/test_persistence.cpython-313-pytest-9.0.3.pyc +0 -0
  39. package/backend/tests/__pycache__/test_workspace_chat.cpython-313-pytest-9.0.3.pyc +0 -0
  40. package/backend/tests/test_agent_tools.py +449 -0
  41. package/backend/tests/test_health.py +12 -0
  42. package/backend/tests/test_llm_providers.py +113 -0
  43. package/backend/tests/test_logging.py +182 -0
  44. package/backend/tests/test_persistence.py +125 -0
  45. package/backend/tests/test_workspace_chat.py +578 -0
  46. package/backend/uv.lock +803 -99
  47. package/dist/frontend/assets/geist-cyrillic-wght-normal-CHSlOQsW.woff2 +0 -0
  48. package/dist/frontend/assets/geist-latin-ext-wght-normal-DMtmJ5ZE.woff2 +0 -0
  49. package/dist/frontend/assets/geist-latin-wght-normal-Dm3htQBi.woff2 +0 -0
  50. package/dist/frontend/assets/index-C76K95ty.js +81 -0
  51. package/dist/frontend/assets/index-iUMNKvlU.css +2 -0
  52. package/dist/frontend/flowent.png +0 -0
  53. package/dist/frontend/index.html +5 -25
  54. package/package.json +1 -2
  55. package/backend/src/flowent/__pycache__/_version.cpython-313.pyc +0 -0
  56. package/backend/src/flowent/__pycache__/access.cpython-313.pyc +0 -0
  57. package/backend/src/flowent/__pycache__/assistant_commands.cpython-313.pyc +0 -0
  58. package/backend/src/flowent/__pycache__/config.cpython-313.pyc +0 -0
  59. package/backend/src/flowent/__pycache__/events.cpython-313.pyc +0 -0
  60. package/backend/src/flowent/__pycache__/graph_runtime.cpython-313.pyc +0 -0
  61. package/backend/src/flowent/__pycache__/graph_service.cpython-313.pyc +0 -0
  62. package/backend/src/flowent/__pycache__/image_assets.cpython-313.pyc +0 -0
  63. package/backend/src/flowent/__pycache__/mcp_service.cpython-313.pyc +0 -0
  64. package/backend/src/flowent/__pycache__/model_metadata.cpython-313.pyc +0 -0
  65. package/backend/src/flowent/__pycache__/network.cpython-313.pyc +0 -0
  66. package/backend/src/flowent/__pycache__/observability_service.cpython-313.pyc +0 -0
  67. package/backend/src/flowent/__pycache__/registry.cpython-313.pyc +0 -0
  68. package/backend/src/flowent/__pycache__/role_management.cpython-313.pyc +0 -0
  69. package/backend/src/flowent/__pycache__/runtime.cpython-313.pyc +0 -0
  70. package/backend/src/flowent/__pycache__/security.cpython-313.pyc +0 -0
  71. package/backend/src/flowent/__pycache__/settings.cpython-313.pyc +0 -0
  72. package/backend/src/flowent/__pycache__/settings_management.cpython-313.pyc +0 -0
  73. package/backend/src/flowent/__pycache__/state_db.cpython-313.pyc +0 -0
  74. package/backend/src/flowent/__pycache__/workspace_store.cpython-313.pyc +0 -0
  75. package/backend/src/flowent/access.py +0 -247
  76. package/backend/src/flowent/assistant_commands.py +0 -115
  77. package/backend/src/flowent/channels/__init__.py +0 -3
  78. package/backend/src/flowent/channels/__pycache__/__init__.cpython-313.pyc +0 -0
  79. package/backend/src/flowent/channels/__pycache__/telegram.cpython-313.pyc +0 -0
  80. package/backend/src/flowent/channels/telegram.py +0 -615
  81. package/backend/src/flowent/config.py +0 -14
  82. package/backend/src/flowent/dev.py +0 -3
  83. package/backend/src/flowent/events.py +0 -157
  84. package/backend/src/flowent/graph_runtime.py +0 -60
  85. package/backend/src/flowent/graph_service.py +0 -2508
  86. package/backend/src/flowent/image_assets.py +0 -356
  87. package/backend/src/flowent/mcp_service.py +0 -1918
  88. package/backend/src/flowent/model_metadata.py +0 -102
  89. package/backend/src/flowent/models/__init__.py +0 -125
  90. package/backend/src/flowent/models/__pycache__/__init__.cpython-313.pyc +0 -0
  91. package/backend/src/flowent/models/__pycache__/agent.cpython-313.pyc +0 -0
  92. package/backend/src/flowent/models/__pycache__/base.cpython-313.pyc +0 -0
  93. package/backend/src/flowent/models/__pycache__/blueprint.cpython-313.pyc +0 -0
  94. package/backend/src/flowent/models/__pycache__/content.cpython-313.pyc +0 -0
  95. package/backend/src/flowent/models/__pycache__/delta.cpython-313.pyc +0 -0
  96. package/backend/src/flowent/models/__pycache__/event.cpython-313.pyc +0 -0
  97. package/backend/src/flowent/models/__pycache__/graph.cpython-313.pyc +0 -0
  98. package/backend/src/flowent/models/__pycache__/history.cpython-313.pyc +0 -0
  99. package/backend/src/flowent/models/__pycache__/llm.cpython-313.pyc +0 -0
  100. package/backend/src/flowent/models/__pycache__/message.cpython-313.pyc +0 -0
  101. package/backend/src/flowent/models/__pycache__/tab.cpython-313.pyc +0 -0
  102. package/backend/src/flowent/models/__pycache__/todo.cpython-313.pyc +0 -0
  103. package/backend/src/flowent/models/agent.py +0 -34
  104. package/backend/src/flowent/models/base.py +0 -24
  105. package/backend/src/flowent/models/blueprint.py +0 -176
  106. package/backend/src/flowent/models/content.py +0 -164
  107. package/backend/src/flowent/models/delta.py +0 -44
  108. package/backend/src/flowent/models/event.py +0 -51
  109. package/backend/src/flowent/models/graph.py +0 -472
  110. package/backend/src/flowent/models/history.py +0 -272
  111. package/backend/src/flowent/models/llm.py +0 -62
  112. package/backend/src/flowent/models/message.py +0 -33
  113. package/backend/src/flowent/models/tab.py +0 -85
  114. package/backend/src/flowent/models/todo.py +0 -10
  115. package/backend/src/flowent/network.py +0 -146
  116. package/backend/src/flowent/observability_service.py +0 -218
  117. package/backend/src/flowent/prompts/__init__.py +0 -67
  118. package/backend/src/flowent/prompts/__pycache__/__init__.cpython-313.pyc +0 -0
  119. package/backend/src/flowent/prompts/__pycache__/common.cpython-313.pyc +0 -0
  120. package/backend/src/flowent/prompts/__pycache__/steward.cpython-313.pyc +0 -0
  121. package/backend/src/flowent/prompts/common.py +0 -250
  122. package/backend/src/flowent/prompts/steward.py +0 -64
  123. package/backend/src/flowent/providers/__init__.py +0 -23
  124. package/backend/src/flowent/providers/__pycache__/__init__.cpython-313.pyc +0 -0
  125. package/backend/src/flowent/providers/__pycache__/anthropic.cpython-313.pyc +0 -0
  126. package/backend/src/flowent/providers/__pycache__/base_url.cpython-313.pyc +0 -0
  127. package/backend/src/flowent/providers/__pycache__/configuration.cpython-313.pyc +0 -0
  128. package/backend/src/flowent/providers/__pycache__/content.cpython-313.pyc +0 -0
  129. package/backend/src/flowent/providers/__pycache__/errors.cpython-313.pyc +0 -0
  130. package/backend/src/flowent/providers/__pycache__/gateway.cpython-313.pyc +0 -0
  131. package/backend/src/flowent/providers/__pycache__/headers.cpython-313.pyc +0 -0
  132. package/backend/src/flowent/providers/__pycache__/management.cpython-313.pyc +0 -0
  133. package/backend/src/flowent/providers/__pycache__/openai.cpython-313.pyc +0 -0
  134. package/backend/src/flowent/providers/__pycache__/openai_responses.cpython-313.pyc +0 -0
  135. package/backend/src/flowent/providers/__pycache__/registry.cpython-313.pyc +0 -0
  136. package/backend/src/flowent/providers/__pycache__/sse.cpython-313.pyc +0 -0
  137. package/backend/src/flowent/providers/__pycache__/thinking.cpython-313.pyc +0 -0
  138. package/backend/src/flowent/providers/anthropic.py +0 -468
  139. package/backend/src/flowent/providers/base_url.py +0 -60
  140. package/backend/src/flowent/providers/configuration.py +0 -189
  141. package/backend/src/flowent/providers/content.py +0 -122
  142. package/backend/src/flowent/providers/errors.py +0 -223
  143. package/backend/src/flowent/providers/gateway.py +0 -169
  144. package/backend/src/flowent/providers/gemini.py +0 -447
  145. package/backend/src/flowent/providers/headers.py +0 -20
  146. package/backend/src/flowent/providers/management.py +0 -96
  147. package/backend/src/flowent/providers/ollama.py +0 -293
  148. package/backend/src/flowent/providers/openai.py +0 -422
  149. package/backend/src/flowent/providers/openai_responses.py +0 -655
  150. package/backend/src/flowent/providers/registry.py +0 -144
  151. package/backend/src/flowent/providers/sse.py +0 -31
  152. package/backend/src/flowent/providers/thinking.py +0 -79
  153. package/backend/src/flowent/registry.py +0 -73
  154. package/backend/src/flowent/role_management.py +0 -267
  155. package/backend/src/flowent/routes/__init__.py +0 -28
  156. package/backend/src/flowent/routes/__pycache__/__init__.cpython-313.pyc +0 -0
  157. package/backend/src/flowent/routes/__pycache__/access.cpython-313.pyc +0 -0
  158. package/backend/src/flowent/routes/__pycache__/assistant.cpython-313.pyc +0 -0
  159. package/backend/src/flowent/routes/__pycache__/image_assets.cpython-313.pyc +0 -0
  160. package/backend/src/flowent/routes/__pycache__/mcp.cpython-313.pyc +0 -0
  161. package/backend/src/flowent/routes/__pycache__/meta.cpython-313.pyc +0 -0
  162. package/backend/src/flowent/routes/__pycache__/nodes.cpython-313.pyc +0 -0
  163. package/backend/src/flowent/routes/__pycache__/prompts.cpython-313.pyc +0 -0
  164. package/backend/src/flowent/routes/__pycache__/providers_route.cpython-313.pyc +0 -0
  165. package/backend/src/flowent/routes/__pycache__/roles.cpython-313.pyc +0 -0
  166. package/backend/src/flowent/routes/__pycache__/settings.cpython-313.pyc +0 -0
  167. package/backend/src/flowent/routes/__pycache__/tabs.cpython-313.pyc +0 -0
  168. package/backend/src/flowent/routes/__pycache__/ws.cpython-313.pyc +0 -0
  169. package/backend/src/flowent/routes/access.py +0 -48
  170. package/backend/src/flowent/routes/assistant.py +0 -155
  171. package/backend/src/flowent/routes/image_assets.py +0 -33
  172. package/backend/src/flowent/routes/mcp.py +0 -125
  173. package/backend/src/flowent/routes/meta.py +0 -28
  174. package/backend/src/flowent/routes/nodes.py +0 -413
  175. package/backend/src/flowent/routes/prompts.py +0 -46
  176. package/backend/src/flowent/routes/providers_route.py +0 -365
  177. package/backend/src/flowent/routes/roles.py +0 -207
  178. package/backend/src/flowent/routes/settings.py +0 -328
  179. package/backend/src/flowent/routes/tabs.py +0 -310
  180. package/backend/src/flowent/routes/ws.py +0 -33
  181. package/backend/src/flowent/runtime.py +0 -165
  182. package/backend/src/flowent/security.py +0 -57
  183. package/backend/src/flowent/settings.py +0 -2518
  184. package/backend/src/flowent/settings_management.py +0 -298
  185. package/backend/src/flowent/state_db.py +0 -120
  186. package/backend/src/flowent/static/assets/AssistantPage-VBohhz4d.js +0 -1
  187. package/backend/src/flowent/static/assets/ChannelsPage-CIydPZA_.js +0 -1
  188. package/backend/src/flowent/static/assets/McpPage-CHPm2TPY.js +0 -7
  189. package/backend/src/flowent/static/assets/PageScaffold-DteOA8V7.js +0 -1
  190. package/backend/src/flowent/static/assets/PromptsPage-CSmJ3sZg.js +0 -1
  191. package/backend/src/flowent/static/assets/ProvidersPage-sl2jeG4e.js +0 -3
  192. package/backend/src/flowent/static/assets/RolesPage-DCe7W6Km.js +0 -1
  193. package/backend/src/flowent/static/assets/SettingsPage-Bix9e63E.js +0 -3
  194. package/backend/src/flowent/static/assets/ToolsPage-favNkj5C.js +0 -1
  195. package/backend/src/flowent/static/assets/WorkspaceCommandDialog-DRS6wiD6.js +0 -1
  196. package/backend/src/flowent/static/assets/WorkspacePage-KuaDjt_D.js +0 -3
  197. package/backend/src/flowent/static/assets/WorkspacePanels-BZxBw8M5.js +0 -1
  198. package/backend/src/flowent/static/assets/alert-dialog-DIBUCmqM.js +0 -1
  199. package/backend/src/flowent/static/assets/datetime-eJqd0V2S.js +0 -1
  200. package/backend/src/flowent/static/assets/dialog-BOvHIBrg.js +0 -1
  201. package/backend/src/flowent/static/assets/elk-worker.min-C9JGDOE-.js +0 -6312
  202. package/backend/src/flowent/static/assets/graph-vendor-CHpVij2M.css +0 -1
  203. package/backend/src/flowent/static/assets/graph-vendor-DRq_-6fV.js +0 -7
  204. package/backend/src/flowent/static/assets/index-Biio-CoI.js +0 -10
  205. package/backend/src/flowent/static/assets/index-CmQvO7sl.css +0 -1
  206. package/backend/src/flowent/static/assets/layout.worker-jMHqAFbP.js +0 -24
  207. package/backend/src/flowent/static/assets/markdown-vendor-C9RtvaJh.js +0 -29
  208. package/backend/src/flowent/static/assets/modelParams-DcEhGnu0.js +0 -1
  209. package/backend/src/flowent/static/assets/react-vendor-mEs_JJxa.js +0 -9
  210. package/backend/src/flowent/static/assets/roles-BbIEIMeG.js +0 -1
  211. package/backend/src/flowent/static/assets/rolldown-runtime-BYbx6iT9.js +0 -1
  212. package/backend/src/flowent/static/assets/select-D9SwnlXF.js +0 -1
  213. package/backend/src/flowent/static/assets/surface-Bzr1FRG4.js +0 -1
  214. package/backend/src/flowent/static/assets/triState-DgLlKdRR.js +0 -1
  215. package/backend/src/flowent/static/assets/ui-vendor-UazN8rcv.js +0 -51
  216. package/backend/src/flowent/static/favicon.svg +0 -4
  217. package/backend/src/flowent/tools/__init__.py +0 -275
  218. package/backend/src/flowent/tools/__pycache__/__init__.cpython-313.pyc +0 -0
  219. package/backend/src/flowent/tools/__pycache__/connect.cpython-313.pyc +0 -0
  220. package/backend/src/flowent/tools/__pycache__/contacts.cpython-313.pyc +0 -0
  221. package/backend/src/flowent/tools/__pycache__/create_agent.cpython-313.pyc +0 -0
  222. package/backend/src/flowent/tools/__pycache__/create_tab.cpython-313.pyc +0 -0
  223. package/backend/src/flowent/tools/__pycache__/delete_tab.cpython-313.pyc +0 -0
  224. package/backend/src/flowent/tools/__pycache__/edit.cpython-313.pyc +0 -0
  225. package/backend/src/flowent/tools/__pycache__/exec.cpython-313.pyc +0 -0
  226. package/backend/src/flowent/tools/__pycache__/fetch.cpython-313.pyc +0 -0
  227. package/backend/src/flowent/tools/__pycache__/idle.cpython-313.pyc +0 -0
  228. package/backend/src/flowent/tools/__pycache__/list_roles.cpython-313.pyc +0 -0
  229. package/backend/src/flowent/tools/__pycache__/list_tabs.cpython-313.pyc +0 -0
  230. package/backend/src/flowent/tools/__pycache__/list_tools.cpython-313.pyc +0 -0
  231. package/backend/src/flowent/tools/__pycache__/manage_prompts.cpython-313.pyc +0 -0
  232. package/backend/src/flowent/tools/__pycache__/manage_providers.cpython-313.pyc +0 -0
  233. package/backend/src/flowent/tools/__pycache__/manage_roles.cpython-313.pyc +0 -0
  234. package/backend/src/flowent/tools/__pycache__/manage_settings.cpython-313.pyc +0 -0
  235. package/backend/src/flowent/tools/__pycache__/mcp.cpython-313.pyc +0 -0
  236. package/backend/src/flowent/tools/__pycache__/read.cpython-313.pyc +0 -0
  237. package/backend/src/flowent/tools/__pycache__/send.cpython-313.pyc +0 -0
  238. package/backend/src/flowent/tools/__pycache__/set_permissions.cpython-313.pyc +0 -0
  239. package/backend/src/flowent/tools/__pycache__/sleep.cpython-313.pyc +0 -0
  240. package/backend/src/flowent/tools/__pycache__/todo.cpython-313.pyc +0 -0
  241. package/backend/src/flowent/tools/connect.py +0 -100
  242. package/backend/src/flowent/tools/contacts.py +0 -22
  243. package/backend/src/flowent/tools/create_agent.py +0 -191
  244. package/backend/src/flowent/tools/create_tab.py +0 -61
  245. package/backend/src/flowent/tools/delete_tab.py +0 -39
  246. package/backend/src/flowent/tools/edit.py +0 -142
  247. package/backend/src/flowent/tools/exec.py +0 -118
  248. package/backend/src/flowent/tools/fetch.py +0 -85
  249. package/backend/src/flowent/tools/idle.py +0 -27
  250. package/backend/src/flowent/tools/list_roles.py +0 -75
  251. package/backend/src/flowent/tools/list_tabs.py +0 -100
  252. package/backend/src/flowent/tools/list_tools.py +0 -28
  253. package/backend/src/flowent/tools/manage_prompts.py +0 -102
  254. package/backend/src/flowent/tools/manage_providers.py +0 -220
  255. package/backend/src/flowent/tools/manage_roles.py +0 -275
  256. package/backend/src/flowent/tools/manage_settings.py +0 -364
  257. package/backend/src/flowent/tools/mcp.py +0 -199
  258. package/backend/src/flowent/tools/read.py +0 -152
  259. package/backend/src/flowent/tools/send.py +0 -68
  260. package/backend/src/flowent/tools/set_permissions.py +0 -99
  261. package/backend/src/flowent/tools/sleep.py +0 -41
  262. package/backend/src/flowent/tools/todo.py +0 -51
  263. package/backend/src/flowent/workspace_store.py +0 -479
  264. package/backend/tests/__init__.py +0 -0
  265. package/backend/tests/__pycache__/__init__.cpython-313.pyc +0 -0
  266. package/backend/tests/__pycache__/conftest.cpython-313-pytest-9.0.3.pyc +0 -0
  267. package/backend/tests/conftest.py +0 -6
  268. package/backend/tests/integration/api/__pycache__/conftest.cpython-313-pytest-9.0.3.pyc +0 -0
  269. package/backend/tests/integration/api/__pycache__/test_access_api.cpython-313-pytest-9.0.3.pyc +0 -0
  270. package/backend/tests/integration/api/__pycache__/test_assistant_api.cpython-313-pytest-9.0.3.pyc +0 -0
  271. package/backend/tests/integration/api/__pycache__/test_frontend_mounting.cpython-313-pytest-9.0.3.pyc +0 -0
  272. package/backend/tests/integration/api/__pycache__/test_mcp_api.cpython-313-pytest-9.0.3.pyc +0 -0
  273. package/backend/tests/integration/api/__pycache__/test_meta_api.cpython-313-pytest-9.0.3.pyc +0 -0
  274. package/backend/tests/integration/api/__pycache__/test_nodes_api.cpython-313-pytest-9.0.3.pyc +0 -0
  275. package/backend/tests/integration/api/__pycache__/test_prompts_api.cpython-313-pytest-9.0.3.pyc +0 -0
  276. package/backend/tests/integration/api/__pycache__/test_roles_api.cpython-313-pytest-9.0.3.pyc +0 -0
  277. package/backend/tests/integration/api/__pycache__/test_tabs_api.cpython-313-pytest-9.0.3.pyc +0 -0
  278. package/backend/tests/integration/api/conftest.py +0 -29
  279. package/backend/tests/integration/api/test_access_api.py +0 -182
  280. package/backend/tests/integration/api/test_assistant_api.py +0 -354
  281. package/backend/tests/integration/api/test_frontend_mounting.py +0 -61
  282. package/backend/tests/integration/api/test_mcp_api.py +0 -116
  283. package/backend/tests/integration/api/test_meta_api.py +0 -33
  284. package/backend/tests/integration/api/test_nodes_api.py +0 -722
  285. package/backend/tests/integration/api/test_prompts_api.py +0 -47
  286. package/backend/tests/integration/api/test_roles_api.py +0 -228
  287. package/backend/tests/integration/api/test_tabs_api.py +0 -802
  288. package/backend/tests/unit/__pycache__/test_access.cpython-313-pytest-9.0.3.pyc +0 -0
  289. package/backend/tests/unit/__pycache__/test_cli.cpython-313-pytest-9.0.3.pyc +0 -0
  290. package/backend/tests/unit/__pycache__/test_graph_runtime.cpython-313-pytest-9.0.3.pyc +0 -0
  291. package/backend/tests/unit/__pycache__/test_network.cpython-313-pytest-9.0.3.pyc +0 -0
  292. package/backend/tests/unit/__pycache__/test_state_sqlite_storage.cpython-313-pytest-9.0.3.pyc +0 -0
  293. package/backend/tests/unit/__pycache__/test_workspace_store.cpython-313-pytest-9.0.3.pyc +0 -0
  294. package/backend/tests/unit/agent/__pycache__/test_agent_public_api.cpython-313-pytest-9.0.3.pyc +0 -0
  295. package/backend/tests/unit/agent/__pycache__/test_agent_runtime.cpython-313-pytest-9.0.3.pyc +0 -0
  296. package/backend/tests/unit/agent/test_agent_public_api.py +0 -837
  297. package/backend/tests/unit/agent/test_agent_runtime.py +0 -2942
  298. package/backend/tests/unit/channels/__pycache__/test_telegram_channel.cpython-313-pytest-9.0.3.pyc +0 -0
  299. package/backend/tests/unit/channels/test_telegram_channel.py +0 -552
  300. package/backend/tests/unit/logging/__pycache__/test_logging.cpython-313-pytest-9.0.3.pyc +0 -0
  301. package/backend/tests/unit/logging/test_logging.py +0 -132
  302. package/backend/tests/unit/prompts/__pycache__/test_prompts.cpython-313-pytest-9.0.3.pyc +0 -0
  303. package/backend/tests/unit/prompts/test_prompts.py +0 -570
  304. package/backend/tests/unit/providers/__pycache__/test_anthropic_provider.cpython-313-pytest-9.0.3.pyc +0 -0
  305. package/backend/tests/unit/providers/__pycache__/test_errors.cpython-313-pytest-9.0.3.pyc +0 -0
  306. package/backend/tests/unit/providers/__pycache__/test_extract_delta_parts.cpython-313-pytest-9.0.3.pyc +0 -0
  307. package/backend/tests/unit/providers/__pycache__/test_openai_provider.cpython-313-pytest-9.0.3.pyc +0 -0
  308. package/backend/tests/unit/providers/__pycache__/test_openai_responses.cpython-313-pytest-9.0.3.pyc +0 -0
  309. package/backend/tests/unit/providers/__pycache__/test_provider_gateway.cpython-313-pytest-9.0.3.pyc +0 -0
  310. package/backend/tests/unit/providers/__pycache__/test_think_tag_parser.cpython-313-pytest-9.0.3.pyc +0 -0
  311. package/backend/tests/unit/providers/test_anthropic_provider.py +0 -185
  312. package/backend/tests/unit/providers/test_errors.py +0 -68
  313. package/backend/tests/unit/providers/test_extract_delta_parts.py +0 -22
  314. package/backend/tests/unit/providers/test_openai_provider.py +0 -139
  315. package/backend/tests/unit/providers/test_openai_responses.py +0 -402
  316. package/backend/tests/unit/providers/test_provider_gateway.py +0 -359
  317. package/backend/tests/unit/providers/test_think_tag_parser.py +0 -36
  318. package/backend/tests/unit/routes/__pycache__/test_prompts_routes.cpython-313-pytest-9.0.3.pyc +0 -0
  319. package/backend/tests/unit/routes/__pycache__/test_providers_route.cpython-313-pytest-9.0.3.pyc +0 -0
  320. package/backend/tests/unit/routes/__pycache__/test_roles_routes.cpython-313-pytest-9.0.3.pyc +0 -0
  321. package/backend/tests/unit/routes/__pycache__/test_settings_routes.cpython-313-pytest-9.0.3.pyc +0 -0
  322. package/backend/tests/unit/routes/test_prompts_routes.py +0 -104
  323. package/backend/tests/unit/routes/test_providers_route.py +0 -370
  324. package/backend/tests/unit/routes/test_roles_routes.py +0 -535
  325. package/backend/tests/unit/routes/test_settings_routes.py +0 -1142
  326. package/backend/tests/unit/runtime/__pycache__/test_bootstrap_runtime.cpython-313-pytest-9.0.3.pyc +0 -0
  327. package/backend/tests/unit/runtime/test_bootstrap_runtime.py +0 -1002
  328. package/backend/tests/unit/sandbox/__pycache__/test_sandbox_tools.cpython-313-pytest-9.0.3.pyc +0 -0
  329. package/backend/tests/unit/sandbox/test_sandbox_tools.py +0 -78
  330. package/backend/tests/unit/security/__pycache__/test_security.cpython-313-pytest-9.0.3.pyc +0 -0
  331. package/backend/tests/unit/security/test_security.py +0 -124
  332. package/backend/tests/unit/settings/__pycache__/test_settings_roles.cpython-313-pytest-9.0.3.pyc +0 -0
  333. package/backend/tests/unit/settings/test_settings_roles.py +0 -751
  334. package/backend/tests/unit/test_access.py +0 -45
  335. package/backend/tests/unit/test_cli.py +0 -124
  336. package/backend/tests/unit/test_graph_runtime.py +0 -72
  337. package/backend/tests/unit/test_network.py +0 -51
  338. package/backend/tests/unit/test_state_sqlite_storage.py +0 -159
  339. package/backend/tests/unit/test_workspace_store.py +0 -231
  340. package/backend/tests/unit/tools/__pycache__/test_connect_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  341. package/backend/tests/unit/tools/__pycache__/test_create_agent_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  342. package/backend/tests/unit/tools/__pycache__/test_delete_tab_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  343. package/backend/tests/unit/tools/__pycache__/test_edit_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  344. package/backend/tests/unit/tools/__pycache__/test_exec_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  345. package/backend/tests/unit/tools/__pycache__/test_fetch_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  346. package/backend/tests/unit/tools/__pycache__/test_manage_prompts_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  347. package/backend/tests/unit/tools/__pycache__/test_manage_providers_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  348. package/backend/tests/unit/tools/__pycache__/test_manage_roles_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  349. package/backend/tests/unit/tools/__pycache__/test_manage_settings_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  350. package/backend/tests/unit/tools/__pycache__/test_read_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  351. package/backend/tests/unit/tools/__pycache__/test_set_permissions_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  352. package/backend/tests/unit/tools/__pycache__/test_todo_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  353. package/backend/tests/unit/tools/__pycache__/test_tool_registry.cpython-313-pytest-9.0.3.pyc +0 -0
  354. package/backend/tests/unit/tools/test_connect_tool.py +0 -228
  355. package/backend/tests/unit/tools/test_create_agent_tool.py +0 -436
  356. package/backend/tests/unit/tools/test_delete_tab_tool.py +0 -116
  357. package/backend/tests/unit/tools/test_edit_tool.py +0 -115
  358. package/backend/tests/unit/tools/test_exec_tool.py +0 -81
  359. package/backend/tests/unit/tools/test_fetch_tool.py +0 -65
  360. package/backend/tests/unit/tools/test_manage_prompts_tool.py +0 -117
  361. package/backend/tests/unit/tools/test_manage_providers_tool.py +0 -460
  362. package/backend/tests/unit/tools/test_manage_roles_tool.py +0 -411
  363. package/backend/tests/unit/tools/test_manage_settings_tool.py +0 -611
  364. package/backend/tests/unit/tools/test_read_tool.py +0 -33
  365. package/backend/tests/unit/tools/test_set_permissions_tool.py +0 -595
  366. package/backend/tests/unit/tools/test_todo_tool.py +0 -37
  367. package/backend/tests/unit/tools/test_tool_registry.py +0 -194
  368. package/dist/frontend/assets/AssistantPage-VBohhz4d.js +0 -1
  369. package/dist/frontend/assets/ChannelsPage-CIydPZA_.js +0 -1
  370. package/dist/frontend/assets/McpPage-CHPm2TPY.js +0 -7
  371. package/dist/frontend/assets/PageScaffold-DteOA8V7.js +0 -1
  372. package/dist/frontend/assets/PromptsPage-CSmJ3sZg.js +0 -1
  373. package/dist/frontend/assets/ProvidersPage-sl2jeG4e.js +0 -3
  374. package/dist/frontend/assets/RolesPage-DCe7W6Km.js +0 -1
  375. package/dist/frontend/assets/SettingsPage-Bix9e63E.js +0 -3
  376. package/dist/frontend/assets/ToolsPage-favNkj5C.js +0 -1
  377. package/dist/frontend/assets/WorkspaceCommandDialog-DRS6wiD6.js +0 -1
  378. package/dist/frontend/assets/WorkspacePage-KuaDjt_D.js +0 -3
  379. package/dist/frontend/assets/WorkspacePanels-BZxBw8M5.js +0 -1
  380. package/dist/frontend/assets/alert-dialog-DIBUCmqM.js +0 -1
  381. package/dist/frontend/assets/datetime-eJqd0V2S.js +0 -1
  382. package/dist/frontend/assets/dialog-BOvHIBrg.js +0 -1
  383. package/dist/frontend/assets/elk-worker.min-C9JGDOE-.js +0 -6312
  384. package/dist/frontend/assets/graph-vendor-CHpVij2M.css +0 -1
  385. package/dist/frontend/assets/graph-vendor-DRq_-6fV.js +0 -7
  386. package/dist/frontend/assets/index-Biio-CoI.js +0 -10
  387. package/dist/frontend/assets/index-CmQvO7sl.css +0 -1
  388. package/dist/frontend/assets/layout.worker-jMHqAFbP.js +0 -24
  389. package/dist/frontend/assets/markdown-vendor-C9RtvaJh.js +0 -29
  390. package/dist/frontend/assets/modelParams-DcEhGnu0.js +0 -1
  391. package/dist/frontend/assets/react-vendor-mEs_JJxa.js +0 -9
  392. package/dist/frontend/assets/roles-BbIEIMeG.js +0 -1
  393. package/dist/frontend/assets/rolldown-runtime-BYbx6iT9.js +0 -1
  394. package/dist/frontend/assets/select-D9SwnlXF.js +0 -1
  395. package/dist/frontend/assets/surface-Bzr1FRG4.js +0 -1
  396. package/dist/frontend/assets/triState-DgLlKdRR.js +0 -1
  397. package/dist/frontend/assets/ui-vendor-UazN8rcv.js +0 -51
  398. package/dist/frontend/favicon.svg +0 -4
@@ -1,595 +0,0 @@
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
- }
@@ -1,37 +0,0 @@
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() == []