flowent 0.0.0 → 0.0.4

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 (494) hide show
  1. package/README.md +70 -10
  2. package/assets/flowent-banner.png +0 -0
  3. package/backend/.python-version +1 -0
  4. package/backend/pyproject.toml +57 -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__/registry.cpython-313.pyc +0 -0
  23. package/backend/src/flowent/__pycache__/role_management.cpython-313.pyc +0 -0
  24. package/backend/src/flowent/__pycache__/runtime.cpython-313.pyc +0 -0
  25. package/backend/src/flowent/__pycache__/sandbox.cpython-313.pyc +0 -0
  26. package/backend/src/flowent/__pycache__/security.cpython-313.pyc +0 -0
  27. package/backend/src/flowent/__pycache__/settings.cpython-313.pyc +0 -0
  28. package/backend/src/flowent/__pycache__/settings_management.cpython-313.pyc +0 -0
  29. package/backend/src/flowent/__pycache__/state_db.cpython-313.pyc +0 -0
  30. package/backend/src/flowent/__pycache__/stats_service.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 +2808 -0
  35. package/backend/src/flowent/assistant_commands.py +106 -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 +1346 -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 +1904 -0
  50. package/backend/src/flowent/model_metadata.py +98 -0
  51. package/backend/src/flowent/models/__init__.py +121 -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 +33 -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 +437 -0
  72. package/backend/src/flowent/models/history.py +214 -0
  73. package/backend/src/flowent/models/llm.py +61 -0
  74. package/backend/src/flowent/models/message.py +27 -0
  75. package/backend/src/flowent/models/tab.py +48 -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/prompts/__init__.py +67 -0
  79. package/backend/src/flowent/prompts/__pycache__/__init__.cpython-313.pyc +0 -0
  80. package/backend/src/flowent/prompts/__pycache__/common.cpython-313.pyc +0 -0
  81. package/backend/src/flowent/prompts/__pycache__/steward.cpython-313.pyc +0 -0
  82. package/backend/src/flowent/prompts/common.py +250 -0
  83. package/backend/src/flowent/prompts/steward.py +64 -0
  84. package/backend/src/flowent/providers/__init__.py +23 -0
  85. package/backend/src/flowent/providers/__pycache__/__init__.cpython-313.pyc +0 -0
  86. package/backend/src/flowent/providers/__pycache__/anthropic.cpython-313.pyc +0 -0
  87. package/backend/src/flowent/providers/__pycache__/base_url.cpython-313.pyc +0 -0
  88. package/backend/src/flowent/providers/__pycache__/configuration.cpython-313.pyc +0 -0
  89. package/backend/src/flowent/providers/__pycache__/content.cpython-313.pyc +0 -0
  90. package/backend/src/flowent/providers/__pycache__/errors.cpython-313.pyc +0 -0
  91. package/backend/src/flowent/providers/__pycache__/gateway.cpython-313.pyc +0 -0
  92. package/backend/src/flowent/providers/__pycache__/headers.cpython-313.pyc +0 -0
  93. package/backend/src/flowent/providers/__pycache__/management.cpython-313.pyc +0 -0
  94. package/backend/src/flowent/providers/__pycache__/openai.cpython-313.pyc +0 -0
  95. package/backend/src/flowent/providers/__pycache__/openai_responses.cpython-313.pyc +0 -0
  96. package/backend/src/flowent/providers/__pycache__/registry.cpython-313.pyc +0 -0
  97. package/backend/src/flowent/providers/__pycache__/sse.cpython-313.pyc +0 -0
  98. package/backend/src/flowent/providers/__pycache__/thinking.cpython-313.pyc +0 -0
  99. package/backend/src/flowent/providers/anthropic.py +468 -0
  100. package/backend/src/flowent/providers/base_url.py +60 -0
  101. package/backend/src/flowent/providers/configuration.py +182 -0
  102. package/backend/src/flowent/providers/content.py +122 -0
  103. package/backend/src/flowent/providers/errors.py +223 -0
  104. package/backend/src/flowent/providers/gateway.py +169 -0
  105. package/backend/src/flowent/providers/gemini.py +447 -0
  106. package/backend/src/flowent/providers/headers.py +20 -0
  107. package/backend/src/flowent/providers/management.py +96 -0
  108. package/backend/src/flowent/providers/ollama.py +293 -0
  109. package/backend/src/flowent/providers/openai.py +422 -0
  110. package/backend/src/flowent/providers/openai_responses.py +655 -0
  111. package/backend/src/flowent/providers/registry.py +144 -0
  112. package/backend/src/flowent/providers/sse.py +31 -0
  113. package/backend/src/flowent/providers/thinking.py +79 -0
  114. package/backend/src/flowent/registry.py +73 -0
  115. package/backend/src/flowent/role_management.py +255 -0
  116. package/backend/src/flowent/routes/__init__.py +30 -0
  117. package/backend/src/flowent/routes/__pycache__/__init__.cpython-313.pyc +0 -0
  118. package/backend/src/flowent/routes/__pycache__/access.cpython-313.pyc +0 -0
  119. package/backend/src/flowent/routes/__pycache__/assistant.cpython-313.pyc +0 -0
  120. package/backend/src/flowent/routes/__pycache__/image_assets.cpython-313.pyc +0 -0
  121. package/backend/src/flowent/routes/__pycache__/mcp.cpython-313.pyc +0 -0
  122. package/backend/src/flowent/routes/__pycache__/meta.cpython-313.pyc +0 -0
  123. package/backend/src/flowent/routes/__pycache__/nodes.cpython-313.pyc +0 -0
  124. package/backend/src/flowent/routes/__pycache__/prompts.cpython-313.pyc +0 -0
  125. package/backend/src/flowent/routes/__pycache__/providers_route.cpython-313.pyc +0 -0
  126. package/backend/src/flowent/routes/__pycache__/roles.cpython-313.pyc +0 -0
  127. package/backend/src/flowent/routes/__pycache__/settings.cpython-313.pyc +0 -0
  128. package/backend/src/flowent/routes/__pycache__/stats.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 +365 -0
  137. package/backend/src/flowent/routes/prompts.py +46 -0
  138. package/backend/src/flowent/routes/providers_route.py +364 -0
  139. package/backend/src/flowent/routes/roles.py +207 -0
  140. package/backend/src/flowent/routes/settings.py +324 -0
  141. package/backend/src/flowent/routes/stats.py +229 -0
  142. package/backend/src/flowent/routes/tabs.py +292 -0
  143. package/backend/src/flowent/routes/ws.py +33 -0
  144. package/backend/src/flowent/runtime.py +188 -0
  145. package/backend/src/flowent/sandbox.py +45 -0
  146. package/backend/src/flowent/security.py +42 -0
  147. package/backend/src/flowent/settings.py +2467 -0
  148. package/backend/src/flowent/settings_management.py +286 -0
  149. package/backend/src/flowent/state_db.py +120 -0
  150. package/backend/src/flowent/static/assets/AssistantPage-B3Xc08AS.js +1 -0
  151. package/backend/src/flowent/static/assets/ChannelsPage-ByLd28xk.js +1 -0
  152. package/backend/src/flowent/static/assets/HomePage-C0hAx9_l.js +3 -0
  153. package/backend/src/flowent/static/assets/McpPage-DkrYLvBv.js +7 -0
  154. package/backend/src/flowent/static/assets/PageScaffold-D4jO9ooX.js +1 -0
  155. package/backend/src/flowent/static/assets/PromptsPage-DWA7rRJd.js +1 -0
  156. package/backend/src/flowent/static/assets/ProvidersPage-PUWT8seJ.js +3 -0
  157. package/backend/src/flowent/static/assets/RolesPage-CqcclGRw.js +1 -0
  158. package/backend/src/flowent/static/assets/SettingsPage-8tS2cJgX.js +3 -0
  159. package/backend/src/flowent/static/assets/StatsPage-BX9khYzu.js +1 -0
  160. package/backend/src/flowent/static/assets/ToolsPage-9Tl9FdeD.js +1 -0
  161. package/backend/src/flowent/static/assets/WorkspaceCommandDialog-CCXxjDL8.js +1 -0
  162. package/backend/src/flowent/static/assets/WorkspacePanels-aMdJ7ZH7.js +1 -0
  163. package/backend/src/flowent/static/assets/alert-dialog-kFYVQ7oX.js +1 -0
  164. package/backend/src/flowent/static/assets/badge-74-3jsCg.js +1 -0
  165. package/backend/src/flowent/static/assets/constants-XUzFf6i1.js +1 -0
  166. package/backend/src/flowent/static/assets/datetime-m6_O_Ci9.js +1 -0
  167. package/backend/src/flowent/static/assets/dialog-BeGSweF6.js +1 -0
  168. package/backend/src/flowent/static/assets/elk-worker.min-C9JGDOE-.js +6312 -0
  169. package/backend/src/flowent/static/assets/graph-vendor-CHpVij2M.css +1 -0
  170. package/backend/src/flowent/static/assets/graph-vendor-DRq_-6fV.js +7 -0
  171. package/backend/src/flowent/static/assets/index-BHC1Vhy8.css +1 -0
  172. package/backend/src/flowent/static/assets/index-CL1ALZ3r.js +10 -0
  173. package/backend/src/flowent/static/assets/layout.worker-jMHqAFbP.js +24 -0
  174. package/backend/src/flowent/static/assets/markdown-vendor-DVdy_w12.js +29 -0
  175. package/backend/src/flowent/static/assets/modelParams-CaHd0903.js +1 -0
  176. package/backend/src/flowent/static/assets/react-vendor-mEs_JJxa.js +9 -0
  177. package/backend/src/flowent/static/assets/roles-2OLDeTc5.js +1 -0
  178. package/backend/src/flowent/static/assets/rolldown-runtime-BYbx6iT9.js +1 -0
  179. package/backend/src/flowent/static/assets/select-DL_LPeDj.js +1 -0
  180. package/backend/src/flowent/static/assets/shared-CMxbpLeQ.js +1 -0
  181. package/backend/src/flowent/static/assets/triState-DEr3NkXV.js +1 -0
  182. package/backend/src/flowent/static/assets/ui-vendor-Dg9NNnWX.js +51 -0
  183. package/backend/src/flowent/static/index.html +36 -0
  184. package/backend/src/flowent/stats_service.py +218 -0
  185. package/backend/src/flowent/tools/__init__.py +201 -0
  186. package/backend/src/flowent/tools/__pycache__/__init__.cpython-313.pyc +0 -0
  187. package/backend/src/flowent/tools/__pycache__/connect.cpython-313.pyc +0 -0
  188. package/backend/src/flowent/tools/__pycache__/contacts.cpython-313.pyc +0 -0
  189. package/backend/src/flowent/tools/__pycache__/create_agent.cpython-313.pyc +0 -0
  190. package/backend/src/flowent/tools/__pycache__/create_tab.cpython-313.pyc +0 -0
  191. package/backend/src/flowent/tools/__pycache__/delete_tab.cpython-313.pyc +0 -0
  192. package/backend/src/flowent/tools/__pycache__/edit.cpython-313.pyc +0 -0
  193. package/backend/src/flowent/tools/__pycache__/exec.cpython-313.pyc +0 -0
  194. package/backend/src/flowent/tools/__pycache__/fetch.cpython-313.pyc +0 -0
  195. package/backend/src/flowent/tools/__pycache__/idle.cpython-313.pyc +0 -0
  196. package/backend/src/flowent/tools/__pycache__/list_roles.cpython-313.pyc +0 -0
  197. package/backend/src/flowent/tools/__pycache__/list_tabs.cpython-313.pyc +0 -0
  198. package/backend/src/flowent/tools/__pycache__/list_tools.cpython-313.pyc +0 -0
  199. package/backend/src/flowent/tools/__pycache__/manage_prompts.cpython-313.pyc +0 -0
  200. package/backend/src/flowent/tools/__pycache__/manage_providers.cpython-313.pyc +0 -0
  201. package/backend/src/flowent/tools/__pycache__/manage_roles.cpython-313.pyc +0 -0
  202. package/backend/src/flowent/tools/__pycache__/manage_settings.cpython-313.pyc +0 -0
  203. package/backend/src/flowent/tools/__pycache__/mcp.cpython-313.pyc +0 -0
  204. package/backend/src/flowent/tools/__pycache__/read.cpython-313.pyc +0 -0
  205. package/backend/src/flowent/tools/__pycache__/send.cpython-313.pyc +0 -0
  206. package/backend/src/flowent/tools/__pycache__/set_permissions.cpython-313.pyc +0 -0
  207. package/backend/src/flowent/tools/__pycache__/sleep.cpython-313.pyc +0 -0
  208. package/backend/src/flowent/tools/__pycache__/todo.cpython-313.pyc +0 -0
  209. package/backend/src/flowent/tools/connect.py +156 -0
  210. package/backend/src/flowent/tools/contacts.py +22 -0
  211. package/backend/src/flowent/tools/create_agent.py +270 -0
  212. package/backend/src/flowent/tools/create_tab.py +59 -0
  213. package/backend/src/flowent/tools/delete_tab.py +39 -0
  214. package/backend/src/flowent/tools/edit.py +142 -0
  215. package/backend/src/flowent/tools/exec.py +117 -0
  216. package/backend/src/flowent/tools/fetch.py +85 -0
  217. package/backend/src/flowent/tools/idle.py +27 -0
  218. package/backend/src/flowent/tools/list_roles.py +50 -0
  219. package/backend/src/flowent/tools/list_tabs.py +96 -0
  220. package/backend/src/flowent/tools/list_tools.py +24 -0
  221. package/backend/src/flowent/tools/manage_prompts.py +102 -0
  222. package/backend/src/flowent/tools/manage_providers.py +220 -0
  223. package/backend/src/flowent/tools/manage_roles.py +275 -0
  224. package/backend/src/flowent/tools/manage_settings.py +346 -0
  225. package/backend/src/flowent/tools/mcp.py +199 -0
  226. package/backend/src/flowent/tools/read.py +152 -0
  227. package/backend/src/flowent/tools/send.py +50 -0
  228. package/backend/src/flowent/tools/set_permissions.py +84 -0
  229. package/backend/src/flowent/tools/sleep.py +41 -0
  230. package/backend/src/flowent/tools/todo.py +51 -0
  231. package/backend/src/flowent/workspace_store.py +479 -0
  232. package/backend/tests/__init__.py +0 -0
  233. package/backend/tests/__pycache__/__init__.cpython-313.pyc +0 -0
  234. package/backend/tests/__pycache__/conftest.cpython-313-pytest-9.0.3.pyc +0 -0
  235. package/backend/tests/conftest.py +6 -0
  236. package/backend/tests/integration/api/__pycache__/conftest.cpython-313-pytest-9.0.3.pyc +0 -0
  237. package/backend/tests/integration/api/__pycache__/test_access_api.cpython-313-pytest-9.0.3.pyc +0 -0
  238. package/backend/tests/integration/api/__pycache__/test_assistant_api.cpython-313-pytest-9.0.3.pyc +0 -0
  239. package/backend/tests/integration/api/__pycache__/test_frontend_mounting.cpython-313-pytest-9.0.3.pyc +0 -0
  240. package/backend/tests/integration/api/__pycache__/test_mcp_api.cpython-313-pytest-9.0.3.pyc +0 -0
  241. package/backend/tests/integration/api/__pycache__/test_meta_api.cpython-313-pytest-9.0.3.pyc +0 -0
  242. package/backend/tests/integration/api/__pycache__/test_nodes_api.cpython-313-pytest-9.0.3.pyc +0 -0
  243. package/backend/tests/integration/api/__pycache__/test_prompts_api.cpython-313-pytest-9.0.3.pyc +0 -0
  244. package/backend/tests/integration/api/__pycache__/test_roles_api.cpython-313-pytest-9.0.3.pyc +0 -0
  245. package/backend/tests/integration/api/__pycache__/test_tabs_api.cpython-313-pytest-9.0.3.pyc +0 -0
  246. package/backend/tests/integration/api/conftest.py +29 -0
  247. package/backend/tests/integration/api/test_access_api.py +182 -0
  248. package/backend/tests/integration/api/test_assistant_api.py +354 -0
  249. package/backend/tests/integration/api/test_frontend_mounting.py +61 -0
  250. package/backend/tests/integration/api/test_mcp_api.py +116 -0
  251. package/backend/tests/integration/api/test_meta_api.py +33 -0
  252. package/backend/tests/integration/api/test_nodes_api.py +486 -0
  253. package/backend/tests/integration/api/test_prompts_api.py +47 -0
  254. package/backend/tests/integration/api/test_roles_api.py +227 -0
  255. package/backend/tests/integration/api/test_tabs_api.py +501 -0
  256. package/backend/tests/unit/__pycache__/test_access.cpython-313-pytest-9.0.3.pyc +0 -0
  257. package/backend/tests/unit/__pycache__/test_cli.cpython-313-pytest-9.0.3.pyc +0 -0
  258. package/backend/tests/unit/__pycache__/test_graph_runtime.cpython-313-pytest-9.0.3.pyc +0 -0
  259. package/backend/tests/unit/__pycache__/test_network.cpython-313-pytest-9.0.3.pyc +0 -0
  260. package/backend/tests/unit/__pycache__/test_state_sqlite_storage.cpython-313-pytest-9.0.3.pyc +0 -0
  261. package/backend/tests/unit/__pycache__/test_workspace_store.cpython-313-pytest-9.0.3.pyc +0 -0
  262. package/backend/tests/unit/agent/__pycache__/test_agent_public_api.cpython-313-pytest-9.0.3.pyc +0 -0
  263. package/backend/tests/unit/agent/__pycache__/test_agent_runtime.cpython-313-pytest-9.0.3.pyc +0 -0
  264. package/backend/tests/unit/agent/test_agent_public_api.py +746 -0
  265. package/backend/tests/unit/agent/test_agent_runtime.py +2726 -0
  266. package/backend/tests/unit/channels/__pycache__/test_telegram_channel.cpython-313-pytest-9.0.3.pyc +0 -0
  267. package/backend/tests/unit/channels/test_telegram_channel.py +552 -0
  268. package/backend/tests/unit/logging/__pycache__/test_logging.cpython-313-pytest-9.0.3.pyc +0 -0
  269. package/backend/tests/unit/logging/test_logging.py +132 -0
  270. package/backend/tests/unit/prompts/__pycache__/test_prompts.cpython-313-pytest-9.0.3.pyc +0 -0
  271. package/backend/tests/unit/prompts/test_prompts.py +569 -0
  272. package/backend/tests/unit/providers/__pycache__/test_anthropic_provider.cpython-313-pytest-9.0.3.pyc +0 -0
  273. package/backend/tests/unit/providers/__pycache__/test_errors.cpython-313-pytest-9.0.3.pyc +0 -0
  274. package/backend/tests/unit/providers/__pycache__/test_extract_delta_parts.cpython-313-pytest-9.0.3.pyc +0 -0
  275. package/backend/tests/unit/providers/__pycache__/test_openai_provider.cpython-313-pytest-9.0.3.pyc +0 -0
  276. package/backend/tests/unit/providers/__pycache__/test_openai_responses.cpython-313-pytest-9.0.3.pyc +0 -0
  277. package/backend/tests/unit/providers/__pycache__/test_provider_gateway.cpython-313-pytest-9.0.3.pyc +0 -0
  278. package/backend/tests/unit/providers/__pycache__/test_think_tag_parser.cpython-313-pytest-9.0.3.pyc +0 -0
  279. package/backend/tests/unit/providers/test_anthropic_provider.py +185 -0
  280. package/backend/tests/unit/providers/test_errors.py +68 -0
  281. package/backend/tests/unit/providers/test_extract_delta_parts.py +22 -0
  282. package/backend/tests/unit/providers/test_openai_provider.py +139 -0
  283. package/backend/tests/unit/providers/test_openai_responses.py +402 -0
  284. package/backend/tests/unit/providers/test_provider_gateway.py +359 -0
  285. package/backend/tests/unit/providers/test_think_tag_parser.py +36 -0
  286. package/backend/tests/unit/routes/__pycache__/test_prompts_routes.cpython-313-pytest-9.0.3.pyc +0 -0
  287. package/backend/tests/unit/routes/__pycache__/test_providers_route.cpython-313-pytest-9.0.3.pyc +0 -0
  288. package/backend/tests/unit/routes/__pycache__/test_roles_routes.cpython-313-pytest-9.0.3.pyc +0 -0
  289. package/backend/tests/unit/routes/__pycache__/test_settings_routes.cpython-313-pytest-9.0.3.pyc +0 -0
  290. package/backend/tests/unit/routes/__pycache__/test_stats_routes.cpython-313-pytest-9.0.3.pyc +0 -0
  291. package/backend/tests/unit/routes/test_prompts_routes.py +104 -0
  292. package/backend/tests/unit/routes/test_providers_route.py +368 -0
  293. package/backend/tests/unit/routes/test_roles_routes.py +426 -0
  294. package/backend/tests/unit/routes/test_settings_routes.py +1138 -0
  295. package/backend/tests/unit/routes/test_stats_routes.py +149 -0
  296. package/backend/tests/unit/runtime/__pycache__/test_bootstrap_runtime.cpython-313-pytest-9.0.3.pyc +0 -0
  297. package/backend/tests/unit/runtime/test_bootstrap_runtime.py +1012 -0
  298. package/backend/tests/unit/sandbox/__pycache__/test_sandbox_tools.cpython-313-pytest-9.0.3.pyc +0 -0
  299. package/backend/tests/unit/sandbox/test_sandbox_tools.py +78 -0
  300. package/backend/tests/unit/security/__pycache__/test_security.cpython-313-pytest-9.0.3.pyc +0 -0
  301. package/backend/tests/unit/security/test_security.py +110 -0
  302. package/backend/tests/unit/settings/__pycache__/test_settings_roles.cpython-313-pytest-9.0.3.pyc +0 -0
  303. package/backend/tests/unit/settings/test_settings_roles.py +711 -0
  304. package/backend/tests/unit/test_access.py +45 -0
  305. package/backend/tests/unit/test_cli.py +124 -0
  306. package/backend/tests/unit/test_graph_runtime.py +72 -0
  307. package/backend/tests/unit/test_network.py +51 -0
  308. package/backend/tests/unit/test_state_sqlite_storage.py +93 -0
  309. package/backend/tests/unit/test_workspace_store.py +231 -0
  310. package/backend/tests/unit/tools/__pycache__/test_connect_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  311. package/backend/tests/unit/tools/__pycache__/test_create_agent_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  312. package/backend/tests/unit/tools/__pycache__/test_delete_tab_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  313. package/backend/tests/unit/tools/__pycache__/test_edit_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  314. package/backend/tests/unit/tools/__pycache__/test_exec_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  315. package/backend/tests/unit/tools/__pycache__/test_fetch_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  316. package/backend/tests/unit/tools/__pycache__/test_manage_prompts_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  317. package/backend/tests/unit/tools/__pycache__/test_manage_providers_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  318. package/backend/tests/unit/tools/__pycache__/test_manage_roles_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  319. package/backend/tests/unit/tools/__pycache__/test_manage_settings_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  320. package/backend/tests/unit/tools/__pycache__/test_read_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  321. package/backend/tests/unit/tools/__pycache__/test_set_permissions_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  322. package/backend/tests/unit/tools/__pycache__/test_todo_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  323. package/backend/tests/unit/tools/__pycache__/test_tool_registry.cpython-313-pytest-9.0.3.pyc +0 -0
  324. package/backend/tests/unit/tools/test_connect_tool.py +229 -0
  325. package/backend/tests/unit/tools/test_create_agent_tool.py +524 -0
  326. package/backend/tests/unit/tools/test_delete_tab_tool.py +83 -0
  327. package/backend/tests/unit/tools/test_edit_tool.py +115 -0
  328. package/backend/tests/unit/tools/test_exec_tool.py +81 -0
  329. package/backend/tests/unit/tools/test_fetch_tool.py +65 -0
  330. package/backend/tests/unit/tools/test_manage_prompts_tool.py +117 -0
  331. package/backend/tests/unit/tools/test_manage_providers_tool.py +458 -0
  332. package/backend/tests/unit/tools/test_manage_roles_tool.py +411 -0
  333. package/backend/tests/unit/tools/test_manage_settings_tool.py +608 -0
  334. package/backend/tests/unit/tools/test_read_tool.py +33 -0
  335. package/backend/tests/unit/tools/test_set_permissions_tool.py +391 -0
  336. package/backend/tests/unit/tools/test_todo_tool.py +37 -0
  337. package/backend/tests/unit/tools/test_tool_registry.py +91 -0
  338. package/backend/uv.lock +1144 -0
  339. package/bin/flowent.mjs +62 -35
  340. package/dist/frontend/assets/AssistantPage-B3Xc08AS.js +1 -0
  341. package/dist/frontend/assets/ChannelsPage-ByLd28xk.js +1 -0
  342. package/dist/frontend/assets/HomePage-C0hAx9_l.js +3 -0
  343. package/dist/frontend/assets/McpPage-DkrYLvBv.js +7 -0
  344. package/dist/frontend/assets/PageScaffold-D4jO9ooX.js +1 -0
  345. package/dist/frontend/assets/PromptsPage-DWA7rRJd.js +1 -0
  346. package/dist/frontend/assets/ProvidersPage-PUWT8seJ.js +3 -0
  347. package/dist/frontend/assets/RolesPage-CqcclGRw.js +1 -0
  348. package/dist/frontend/assets/SettingsPage-8tS2cJgX.js +3 -0
  349. package/dist/frontend/assets/StatsPage-BX9khYzu.js +1 -0
  350. package/dist/frontend/assets/ToolsPage-9Tl9FdeD.js +1 -0
  351. package/dist/frontend/assets/WorkspaceCommandDialog-CCXxjDL8.js +1 -0
  352. package/dist/frontend/assets/WorkspacePanels-aMdJ7ZH7.js +1 -0
  353. package/dist/frontend/assets/alert-dialog-kFYVQ7oX.js +1 -0
  354. package/dist/frontend/assets/badge-74-3jsCg.js +1 -0
  355. package/dist/frontend/assets/constants-XUzFf6i1.js +1 -0
  356. package/dist/frontend/assets/datetime-m6_O_Ci9.js +1 -0
  357. package/dist/frontend/assets/dialog-BeGSweF6.js +1 -0
  358. package/dist/frontend/assets/elk-worker.min-C9JGDOE-.js +6312 -0
  359. package/dist/frontend/assets/graph-vendor-CHpVij2M.css +1 -0
  360. package/dist/frontend/assets/graph-vendor-DRq_-6fV.js +7 -0
  361. package/dist/frontend/assets/index-BHC1Vhy8.css +1 -0
  362. package/dist/frontend/assets/index-CL1ALZ3r.js +10 -0
  363. package/dist/frontend/assets/layout.worker-jMHqAFbP.js +24 -0
  364. package/dist/frontend/assets/markdown-vendor-DVdy_w12.js +29 -0
  365. package/dist/frontend/assets/modelParams-CaHd0903.js +1 -0
  366. package/dist/frontend/assets/react-vendor-mEs_JJxa.js +9 -0
  367. package/dist/frontend/assets/roles-2OLDeTc5.js +1 -0
  368. package/dist/frontend/assets/rolldown-runtime-BYbx6iT9.js +1 -0
  369. package/dist/frontend/assets/select-DL_LPeDj.js +1 -0
  370. package/dist/frontend/assets/shared-CMxbpLeQ.js +1 -0
  371. package/dist/frontend/assets/triState-DEr3NkXV.js +1 -0
  372. package/dist/frontend/assets/ui-vendor-Dg9NNnWX.js +51 -0
  373. package/dist/frontend/index.html +36 -0
  374. package/package.json +28 -41
  375. package/dist/.next/BUILD_ID +0 -1
  376. package/dist/.next/app-path-routes-manifest.json +0 -6
  377. package/dist/.next/build-manifest.json +0 -20
  378. package/dist/.next/package.json +0 -1
  379. package/dist/.next/prerender-manifest.json +0 -114
  380. package/dist/.next/required-server-files.json +0 -333
  381. package/dist/.next/routes-manifest.json +0 -69
  382. package/dist/.next/server/app/_global-error/page/app-paths-manifest.json +0 -3
  383. package/dist/.next/server/app/_global-error/page/build-manifest.json +0 -16
  384. package/dist/.next/server/app/_global-error/page/next-font-manifest.json +0 -6
  385. package/dist/.next/server/app/_global-error/page/react-loadable-manifest.json +0 -1
  386. package/dist/.next/server/app/_global-error/page/server-reference-manifest.json +0 -4
  387. package/dist/.next/server/app/_global-error/page.js +0 -9
  388. package/dist/.next/server/app/_global-error/page.js.map +0 -5
  389. package/dist/.next/server/app/_global-error/page.js.nft.json +0 -1
  390. package/dist/.next/server/app/_global-error/page_client-reference-manifest.js +0 -3
  391. package/dist/.next/server/app/_global-error.html +0 -1
  392. package/dist/.next/server/app/_global-error.meta +0 -15
  393. package/dist/.next/server/app/_global-error.rsc +0 -14
  394. package/dist/.next/server/app/_global-error.segments/__PAGE__.segment.rsc +0 -5
  395. package/dist/.next/server/app/_global-error.segments/_full.segment.rsc +0 -14
  396. package/dist/.next/server/app/_global-error.segments/_head.segment.rsc +0 -5
  397. package/dist/.next/server/app/_global-error.segments/_index.segment.rsc +0 -5
  398. package/dist/.next/server/app/_global-error.segments/_tree.segment.rsc +0 -1
  399. package/dist/.next/server/app/_not-found/page/app-paths-manifest.json +0 -3
  400. package/dist/.next/server/app/_not-found/page/build-manifest.json +0 -16
  401. package/dist/.next/server/app/_not-found/page/next-font-manifest.json +0 -10
  402. package/dist/.next/server/app/_not-found/page/react-loadable-manifest.json +0 -1
  403. package/dist/.next/server/app/_not-found/page/server-reference-manifest.json +0 -4
  404. package/dist/.next/server/app/_not-found/page.js +0 -13
  405. package/dist/.next/server/app/_not-found/page.js.map +0 -5
  406. package/dist/.next/server/app/_not-found/page.js.nft.json +0 -1
  407. package/dist/.next/server/app/_not-found/page_client-reference-manifest.js +0 -3
  408. package/dist/.next/server/app/_not-found.html +0 -1
  409. package/dist/.next/server/app/_not-found.meta +0 -16
  410. package/dist/.next/server/app/_not-found.rsc +0 -16
  411. package/dist/.next/server/app/_not-found.segments/_full.segment.rsc +0 -16
  412. package/dist/.next/server/app/_not-found.segments/_head.segment.rsc +0 -6
  413. package/dist/.next/server/app/_not-found.segments/_index.segment.rsc +0 -5
  414. package/dist/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +0 -5
  415. package/dist/.next/server/app/_not-found.segments/_not-found.segment.rsc +0 -5
  416. package/dist/.next/server/app/_not-found.segments/_tree.segment.rsc +0 -2
  417. package/dist/.next/server/app/icon.svg/route/app-paths-manifest.json +0 -3
  418. package/dist/.next/server/app/icon.svg/route/build-manifest.json +0 -9
  419. package/dist/.next/server/app/icon.svg/route.js +0 -6
  420. package/dist/.next/server/app/icon.svg/route.js.map +0 -5
  421. package/dist/.next/server/app/icon.svg/route.js.nft.json +0 -1
  422. package/dist/.next/server/app/icon.svg.meta +0 -1
  423. package/dist/.next/server/app/index.html +0 -1
  424. package/dist/.next/server/app/index.meta +0 -14
  425. package/dist/.next/server/app/index.rsc +0 -15
  426. package/dist/.next/server/app/index.segments/__PAGE__.segment.rsc +0 -5
  427. package/dist/.next/server/app/index.segments/_full.segment.rsc +0 -15
  428. package/dist/.next/server/app/index.segments/_head.segment.rsc +0 -6
  429. package/dist/.next/server/app/index.segments/_index.segment.rsc +0 -5
  430. package/dist/.next/server/app/index.segments/_tree.segment.rsc +0 -3
  431. package/dist/.next/server/app/page/app-paths-manifest.json +0 -3
  432. package/dist/.next/server/app/page/build-manifest.json +0 -16
  433. package/dist/.next/server/app/page/next-font-manifest.json +0 -10
  434. package/dist/.next/server/app/page/react-loadable-manifest.json +0 -1
  435. package/dist/.next/server/app/page/server-reference-manifest.json +0 -4
  436. package/dist/.next/server/app/page.js +0 -14
  437. package/dist/.next/server/app/page.js.map +0 -5
  438. package/dist/.next/server/app/page.js.nft.json +0 -1
  439. package/dist/.next/server/app/page_client-reference-manifest.js +0 -3
  440. package/dist/.next/server/app-paths-manifest.json +0 -6
  441. package/dist/.next/server/chunks/[externals]_next_dist_0arv.vj._.js +0 -3
  442. package/dist/.next/server/chunks/[root-of-the-server]__0vcj1q1._.js +0 -13
  443. package/dist/.next/server/chunks/[turbopack]_runtime.js +0 -903
  444. package/dist/.next/server/chunks/_next-internal_server_app_icon_svg_route_actions_0-0ehc~.js +0 -3
  445. package/dist/.next/server/chunks/ssr/05w9_next_dist_0ihu0u9._.js +0 -6
  446. package/dist/.next/server/chunks/ssr/05w9_next_dist_client_components_12u3mib._.js +0 -3
  447. package/dist/.next/server/chunks/ssr/05w9_next_dist_client_components_builtin_forbidden_04fbe_..js +0 -3
  448. package/dist/.next/server/chunks/ssr/05w9_next_dist_client_components_builtin_global-error_0brpl_..js +0 -3
  449. package/dist/.next/server/chunks/ssr/05w9_next_dist_client_components_builtin_unauthorized_0~2g66g.js +0 -3
  450. package/dist/.next/server/chunks/ssr/05w9_next_dist_esm_build_templates_app-page_0~cyr1_.js +0 -4
  451. package/dist/.next/server/chunks/ssr/05w9_next_dist_esm_build_templates_app-page_1105emf.js +0 -4
  452. package/dist/.next/server/chunks/ssr/05w9_next_dist_esm_build_templates_app-page_11uhyqv.js +0 -4
  453. package/dist/.next/server/chunks/ssr/[root-of-the-server]__0.t9_75._.js +0 -33
  454. package/dist/.next/server/chunks/ssr/[root-of-the-server]__0c0ud_z._.js +0 -3
  455. package/dist/.next/server/chunks/ssr/[root-of-the-server]__0f9_8d4._.js +0 -3
  456. package/dist/.next/server/chunks/ssr/[root-of-the-server]__0l5ko41._.js +0 -19
  457. package/dist/.next/server/chunks/ssr/[root-of-the-server]__0mn6z7i._.js +0 -3
  458. package/dist/.next/server/chunks/ssr/[root-of-the-server]__0npxxst._.js +0 -33
  459. package/dist/.next/server/chunks/ssr/[root-of-the-server]__0qjhaca._.js +0 -3
  460. package/dist/.next/server/chunks/ssr/[root-of-the-server]__0rwgw3s._.js +0 -3
  461. package/dist/.next/server/chunks/ssr/[turbopack]_runtime.js +0 -903
  462. package/dist/.next/server/chunks/ssr/_next-internal_server_app__global-error_page_actions_0k77kol.js +0 -3
  463. package/dist/.next/server/chunks/ssr/_next-internal_server_app__not-found_page_actions_0eq97pa.js +0 -3
  464. package/dist/.next/server/chunks/ssr/_next-internal_server_app_page_actions_09-gtaw.js +0 -3
  465. package/dist/.next/server/chunks/ssr/node_modules__pnpm_056~6.6._.js +0 -3
  466. package/dist/.next/server/chunks/ssr/node_modules__pnpm_0~j0k.e._.js +0 -33
  467. package/dist/.next/server/functions-config-manifest.json +0 -4
  468. package/dist/.next/server/middleware-build-manifest.js +0 -20
  469. package/dist/.next/server/middleware-manifest.json +0 -6
  470. package/dist/.next/server/next-font-manifest.js +0 -1
  471. package/dist/.next/server/next-font-manifest.json +0 -13
  472. package/dist/.next/server/pages/404.html +0 -1
  473. package/dist/.next/server/pages/500.html +0 -1
  474. package/dist/.next/server/pages-manifest.json +0 -4
  475. package/dist/.next/server/prefetch-hints.json +0 -1
  476. package/dist/.next/server/server-reference-manifest.js +0 -1
  477. package/dist/.next/server/server-reference-manifest.json +0 -5
  478. package/dist/.next/static/7FFlzRe2eS-D0Lw5oEpmC/_buildManifest.js +0 -11
  479. package/dist/.next/static/7FFlzRe2eS-D0Lw5oEpmC/_clientMiddlewareManifest.js +0 -1
  480. package/dist/.next/static/7FFlzRe2eS-D0Lw5oEpmC/_ssgManifest.js +0 -1
  481. package/dist/.next/static/chunks/01qk2~bgf76vu.js +0 -1
  482. package/dist/.next/static/chunks/03~yq9q893hmn.js +0 -1
  483. package/dist/.next/static/chunks/080queev.r2uy.js +0 -31
  484. package/dist/.next/static/chunks/0v3lyuj75aq50.js +0 -1
  485. package/dist/.next/static/chunks/10b~xdx5c-i7s.js +0 -5
  486. package/dist/.next/static/chunks/15~9l5n.~r-.4.css +0 -2
  487. package/dist/.next/static/chunks/turbopack-0m-970~qvs7sc.js +0 -1
  488. package/dist/.next/static/media/7178b3e590c64307-s.11.cyxs5p-0z~.woff2 +0 -0
  489. package/dist/.next/static/media/8a480f0b521d4e75-s.06d3mdzz5bre_.woff2 +0 -0
  490. package/dist/.next/static/media/caa3a2e1cccd8315-s.p.16t1db8_9y2o~.woff2 +0 -0
  491. package/dist/package.json +0 -87
  492. package/dist/server.js +0 -38
  493. /package/{dist/.next/server/app/icon.svg.body → backend/src/flowent/static/favicon.svg} +0 -0
  494. /package/dist/{.next/static/media/icon.0.r~afrtrocz9.svg → frontend/favicon.svg} +0 -0
@@ -0,0 +1,1012 @@
1
+ import json
2
+ from uuid import UUID
3
+
4
+ import pytest
5
+
6
+ import flowent.settings as settings_module
7
+ from flowent.agent import Agent
8
+ from flowent.models import AgentState, AssistantText, ReceivedMessage, StateEntry
9
+ from flowent.registry import registry
10
+ from flowent.runtime import bootstrap_runtime, shutdown_runtime
11
+ from flowent.settings import (
12
+ CONDUCTOR_ROLE_DESCRIPTION,
13
+ CONDUCTOR_ROLE_INCLUDED_TOOLS,
14
+ CONDUCTOR_ROLE_NAME,
15
+ CONDUCTOR_ROLE_SYSTEM_PROMPT,
16
+ DESIGNER_ROLE_DESCRIPTION,
17
+ DESIGNER_ROLE_INCLUDED_TOOLS,
18
+ DESIGNER_ROLE_NAME,
19
+ DESIGNER_ROLE_SYSTEM_PROMPT,
20
+ STEWARD_ROLE_DESCRIPTION,
21
+ STEWARD_ROLE_INCLUDED_TOOLS,
22
+ STEWARD_ROLE_NAME,
23
+ STEWARD_ROLE_SYSTEM_PROMPT,
24
+ WORKER_ROLE_DESCRIPTION,
25
+ WORKER_ROLE_INCLUDED_TOOLS,
26
+ WORKER_ROLE_NAME,
27
+ WORKER_ROLE_SYSTEM_PROMPT,
28
+ RoleConfig,
29
+ build_default_assistant_write_dirs,
30
+ )
31
+ from flowent.tools import MINIMUM_TOOLS
32
+ from flowent.workspace_store import workspace_store
33
+
34
+
35
+ def test_bootstrap_runtime_creates_only_assistant(
36
+ monkeypatch,
37
+ tmp_path,
38
+ ):
39
+ registry.reset()
40
+ settings_file = tmp_path / "settings.json"
41
+ settings_file.write_text(
42
+ json.dumps(
43
+ {
44
+ "event_log": {"timestamp_format": "absolute"},
45
+ "model": {"active_provider_id": "", "active_model": ""},
46
+ "providers": [],
47
+ "roles": [],
48
+ }
49
+ ),
50
+ encoding="utf-8",
51
+ )
52
+
53
+ monkeypatch.setattr(Agent, "start", lambda self: None)
54
+ monkeypatch.setattr(settings_module, "_SETTINGS_FILE", settings_file)
55
+ monkeypatch.setattr(settings_module, "_cached_settings", None)
56
+
57
+ bootstrap_runtime()
58
+
59
+ try:
60
+ nodes = registry.get_all()
61
+ assistant = registry.get_assistant()
62
+
63
+ assert len(nodes) == 1
64
+ assert assistant is not None
65
+ assert str(UUID(assistant.uuid)) == assistant.uuid
66
+ assert assistant.config.node_type.value == "assistant"
67
+ assert assistant.config.name == "Assistant"
68
+ assert assistant.config.role_name == STEWARD_ROLE_NAME
69
+ assert set(MINIMUM_TOOLS).issubset(set(assistant.config.tools))
70
+ assert set(STEWARD_ROLE_INCLUDED_TOOLS).issubset(set(assistant.config.tools))
71
+ assert assistant.config.write_dirs == build_default_assistant_write_dirs()
72
+ assert assistant.config.allow_network is True
73
+ finally:
74
+ registry.reset()
75
+
76
+
77
+ def test_bootstrap_runtime_restores_assistant_history(monkeypatch, tmp_path):
78
+ settings_file = tmp_path / "settings.json"
79
+ workspace_file = tmp_path / "workspace.json"
80
+ assistant_id = "00000000-0000-0000-0000-000000000001"
81
+ settings_file.write_text(
82
+ json.dumps(
83
+ {
84
+ "event_log": {"timestamp_format": "absolute"},
85
+ "model": {"active_provider_id": "", "active_model": ""},
86
+ "providers": [],
87
+ "roles": [],
88
+ }
89
+ ),
90
+ encoding="utf-8",
91
+ )
92
+ workspace_file.write_text(
93
+ json.dumps(
94
+ {
95
+ "tabs": [],
96
+ "nodes": [
97
+ {
98
+ "id": assistant_id,
99
+ "config": {
100
+ "node_type": "assistant",
101
+ "role_name": "Steward",
102
+ "tab_id": None,
103
+ "name": "Assistant",
104
+ "tools": [],
105
+ "write_dirs": [str(tmp_path)],
106
+ "allow_network": True,
107
+ },
108
+ "state": "idle",
109
+ "todos": [],
110
+ "history": [
111
+ {
112
+ "type": "ReceivedMessage",
113
+ "content": "hello",
114
+ "from_id": "human",
115
+ "timestamp": 1,
116
+ },
117
+ {
118
+ "type": "AssistantText",
119
+ "content": "hi there",
120
+ "timestamp": 2,
121
+ },
122
+ ],
123
+ "position": None,
124
+ "created_at": 1,
125
+ "updated_at": 1,
126
+ }
127
+ ],
128
+ "edges": [],
129
+ }
130
+ ),
131
+ encoding="utf-8",
132
+ )
133
+
134
+ monkeypatch.setattr(Agent, "start", lambda self: None)
135
+ monkeypatch.setattr(settings_module, "_SETTINGS_FILE", settings_file)
136
+ monkeypatch.setattr(settings_module, "_cached_settings", None)
137
+
138
+ bootstrap_runtime()
139
+
140
+ try:
141
+ assistant = registry.get_assistant()
142
+ assert assistant is not None
143
+ assert assistant.uuid == assistant_id
144
+ assert assistant.state == AgentState.IDLE
145
+ assert any(
146
+ isinstance(entry, ReceivedMessage) and entry.content == "hello"
147
+ for entry in assistant.history
148
+ )
149
+ assert any(
150
+ isinstance(entry, AssistantText) and entry.content == "hi there"
151
+ for entry in assistant.history
152
+ )
153
+ finally:
154
+ registry.reset()
155
+
156
+
157
+ def _stop_all_agents(timeout: float = 1.0) -> None:
158
+ for node in list(registry.get_all()):
159
+ node.request_termination("test cleanup")
160
+ for node in list(registry.get_all()):
161
+ node.wait_for_termination(timeout=timeout)
162
+ registry.reset()
163
+
164
+
165
+ def test_bootstrap_runtime_creates_builtin_roles(
166
+ monkeypatch,
167
+ tmp_path,
168
+ ):
169
+ registry.reset()
170
+ settings_file = tmp_path / "settings.json"
171
+ settings_file.write_text(
172
+ json.dumps(
173
+ {
174
+ "event_log": {"timestamp_format": "absolute"},
175
+ "model": {"active_provider_id": "", "active_model": ""},
176
+ "providers": [],
177
+ "roles": [],
178
+ }
179
+ ),
180
+ encoding="utf-8",
181
+ )
182
+
183
+ monkeypatch.setattr(Agent, "start", lambda self: None)
184
+ monkeypatch.setattr(settings_module, "_SETTINGS_FILE", settings_file)
185
+ monkeypatch.setattr(settings_module, "_cached_settings", None)
186
+
187
+ bootstrap_runtime()
188
+
189
+ try:
190
+ settings = settings_module.get_settings()
191
+
192
+ assert settings.roles == [
193
+ RoleConfig(
194
+ name=STEWARD_ROLE_NAME,
195
+ description=STEWARD_ROLE_DESCRIPTION,
196
+ system_prompt=STEWARD_ROLE_SYSTEM_PROMPT,
197
+ included_tools=STEWARD_ROLE_INCLUDED_TOOLS,
198
+ ),
199
+ RoleConfig(
200
+ name=WORKER_ROLE_NAME,
201
+ description=WORKER_ROLE_DESCRIPTION,
202
+ system_prompt=WORKER_ROLE_SYSTEM_PROMPT,
203
+ included_tools=WORKER_ROLE_INCLUDED_TOOLS,
204
+ ),
205
+ RoleConfig(
206
+ name=CONDUCTOR_ROLE_NAME,
207
+ description=CONDUCTOR_ROLE_DESCRIPTION,
208
+ system_prompt=CONDUCTOR_ROLE_SYSTEM_PROMPT,
209
+ included_tools=CONDUCTOR_ROLE_INCLUDED_TOOLS,
210
+ ),
211
+ RoleConfig(
212
+ name=DESIGNER_ROLE_NAME,
213
+ description=DESIGNER_ROLE_DESCRIPTION,
214
+ system_prompt=DESIGNER_ROLE_SYSTEM_PROMPT,
215
+ included_tools=DESIGNER_ROLE_INCLUDED_TOOLS,
216
+ ),
217
+ ]
218
+ finally:
219
+ registry.reset()
220
+
221
+
222
+ def test_bootstrap_runtime_reconciles_existing_builtin_roles(monkeypatch, tmp_path):
223
+ registry.reset()
224
+ settings_file = tmp_path / "settings.json"
225
+ settings_file.write_text(
226
+ json.dumps(
227
+ {
228
+ "event_log": {"timestamp_format": "absolute"},
229
+ "model": {"active_provider_id": "", "active_model": ""},
230
+ "providers": [],
231
+ "roles": [
232
+ {
233
+ "name": WORKER_ROLE_NAME,
234
+ "system_prompt": "Custom worker prompt.",
235
+ "included_tools": [],
236
+ "excluded_tools": ["fetch"],
237
+ }
238
+ ],
239
+ }
240
+ ),
241
+ encoding="utf-8",
242
+ )
243
+
244
+ monkeypatch.setattr(Agent, "start", lambda self: None)
245
+ monkeypatch.setattr(settings_module, "_SETTINGS_FILE", settings_file)
246
+ monkeypatch.setattr(settings_module, "_cached_settings", None)
247
+
248
+ bootstrap_runtime()
249
+
250
+ try:
251
+ settings = settings_module.get_settings()
252
+
253
+ assert settings.roles == [
254
+ RoleConfig(
255
+ name=STEWARD_ROLE_NAME,
256
+ description=STEWARD_ROLE_DESCRIPTION,
257
+ system_prompt=STEWARD_ROLE_SYSTEM_PROMPT,
258
+ included_tools=STEWARD_ROLE_INCLUDED_TOOLS,
259
+ ),
260
+ RoleConfig(
261
+ name=WORKER_ROLE_NAME,
262
+ description=WORKER_ROLE_DESCRIPTION,
263
+ system_prompt=WORKER_ROLE_SYSTEM_PROMPT,
264
+ included_tools=WORKER_ROLE_INCLUDED_TOOLS,
265
+ ),
266
+ RoleConfig(
267
+ name=CONDUCTOR_ROLE_NAME,
268
+ description=CONDUCTOR_ROLE_DESCRIPTION,
269
+ system_prompt=CONDUCTOR_ROLE_SYSTEM_PROMPT,
270
+ included_tools=CONDUCTOR_ROLE_INCLUDED_TOOLS,
271
+ ),
272
+ RoleConfig(
273
+ name=DESIGNER_ROLE_NAME,
274
+ description=DESIGNER_ROLE_DESCRIPTION,
275
+ system_prompt=DESIGNER_ROLE_SYSTEM_PROMPT,
276
+ included_tools=DESIGNER_ROLE_INCLUDED_TOOLS,
277
+ ),
278
+ ]
279
+ finally:
280
+ registry.reset()
281
+
282
+
283
+ def test_bootstrap_runtime_uses_configured_assistant_role(monkeypatch, tmp_path):
284
+ registry.reset()
285
+ settings_file = tmp_path / "settings.json"
286
+ settings_file.write_text(
287
+ json.dumps(
288
+ {
289
+ "assistant": {"role_name": "Reviewer"},
290
+ "event_log": {"timestamp_format": "absolute"},
291
+ "model": {"active_provider_id": "", "active_model": ""},
292
+ "providers": [],
293
+ "roles": [
294
+ {
295
+ "name": "Reviewer",
296
+ "system_prompt": "Review everything.",
297
+ "included_tools": [],
298
+ "excluded_tools": [],
299
+ }
300
+ ],
301
+ }
302
+ ),
303
+ encoding="utf-8",
304
+ )
305
+
306
+ monkeypatch.setattr(Agent, "start", lambda self: None)
307
+ monkeypatch.setattr(settings_module, "_SETTINGS_FILE", settings_file)
308
+ monkeypatch.setattr(settings_module, "_cached_settings", None)
309
+
310
+ bootstrap_runtime()
311
+
312
+ try:
313
+ assistant = registry.get_assistant()
314
+ assert assistant is not None
315
+ assert assistant.config.role_name == "Reviewer"
316
+ assert settings_module.get_settings().assistant.role_name == "Reviewer"
317
+ finally:
318
+ registry.reset()
319
+
320
+
321
+ def test_bootstrap_runtime_preserves_steward_tools_for_non_steward_assistant_role(
322
+ monkeypatch,
323
+ tmp_path,
324
+ ):
325
+ registry.reset()
326
+ settings_file = tmp_path / "settings.json"
327
+ settings_file.write_text(
328
+ json.dumps(
329
+ {
330
+ "assistant": {"role_name": WORKER_ROLE_NAME},
331
+ "event_log": {"timestamp_format": "absolute"},
332
+ "model": {"active_provider_id": "", "active_model": ""},
333
+ "providers": [],
334
+ "roles": [],
335
+ }
336
+ ),
337
+ encoding="utf-8",
338
+ )
339
+
340
+ monkeypatch.setattr(Agent, "start", lambda self: None)
341
+ monkeypatch.setattr(settings_module, "_SETTINGS_FILE", settings_file)
342
+ monkeypatch.setattr(settings_module, "_cached_settings", None)
343
+
344
+ bootstrap_runtime()
345
+
346
+ try:
347
+ assistant = registry.get_assistant()
348
+ assert assistant is not None
349
+ assert assistant.config.role_name == WORKER_ROLE_NAME
350
+ assert "create_workflow" in assistant.config.tools
351
+ assert "delete_workflow" in assistant.config.tools
352
+ assert "set_permissions" in assistant.config.tools
353
+ assert "manage_settings" in assistant.config.tools
354
+ assert "read" in assistant.config.tools
355
+ assert "exec" in assistant.config.tools
356
+ finally:
357
+ registry.reset()
358
+
359
+
360
+ def test_bootstrap_runtime_uses_configured_assistant_permissions(
361
+ monkeypatch,
362
+ tmp_path,
363
+ ):
364
+ registry.reset()
365
+ allowed_dir = tmp_path / "assistant-write"
366
+ settings_file = tmp_path / "settings.json"
367
+ settings_file.write_text(
368
+ json.dumps(
369
+ {
370
+ "assistant": {
371
+ "role_name": "Reviewer",
372
+ "allow_network": False,
373
+ "write_dirs": [str(allowed_dir)],
374
+ },
375
+ "event_log": {"timestamp_format": "absolute"},
376
+ "model": {"active_provider_id": "", "active_model": ""},
377
+ "providers": [],
378
+ "roles": [
379
+ {
380
+ "name": "Reviewer",
381
+ "system_prompt": "Review everything.",
382
+ "included_tools": [],
383
+ "excluded_tools": [],
384
+ }
385
+ ],
386
+ }
387
+ ),
388
+ encoding="utf-8",
389
+ )
390
+
391
+ monkeypatch.setattr(Agent, "start", lambda self: None)
392
+ monkeypatch.setattr(settings_module, "_SETTINGS_FILE", settings_file)
393
+ monkeypatch.setattr(settings_module, "_cached_settings", None)
394
+
395
+ bootstrap_runtime()
396
+
397
+ try:
398
+ assistant = registry.get_assistant()
399
+ assert assistant is not None
400
+ assert assistant.config.allow_network is False
401
+ assert assistant.config.write_dirs == [str(allowed_dir.resolve())]
402
+ finally:
403
+ registry.reset()
404
+
405
+
406
+ def test_bootstrap_runtime_falls_back_to_steward_when_assistant_role_missing(
407
+ monkeypatch,
408
+ tmp_path,
409
+ ):
410
+ registry.reset()
411
+ settings_file = tmp_path / "settings.json"
412
+ settings_file.write_text(
413
+ json.dumps(
414
+ {
415
+ "assistant": {"role_name": "Ghost"},
416
+ "event_log": {"timestamp_format": "absolute"},
417
+ "model": {"active_provider_id": "", "active_model": ""},
418
+ "providers": [],
419
+ "roles": [],
420
+ }
421
+ ),
422
+ encoding="utf-8",
423
+ )
424
+
425
+ monkeypatch.setattr(Agent, "start", lambda self: None)
426
+ monkeypatch.setattr(settings_module, "_SETTINGS_FILE", settings_file)
427
+ monkeypatch.setattr(settings_module, "_cached_settings", None)
428
+
429
+ bootstrap_runtime()
430
+
431
+ try:
432
+ assistant = registry.get_assistant()
433
+ assert assistant is not None
434
+ assert assistant.config.role_name == STEWARD_ROLE_NAME
435
+ assert settings_module.get_settings().assistant.role_name == STEWARD_ROLE_NAME
436
+ finally:
437
+ registry.reset()
438
+
439
+
440
+ def test_bootstrap_runtime_backfills_state_history_for_restored_nodes(
441
+ monkeypatch,
442
+ tmp_path,
443
+ ):
444
+ registry.reset()
445
+ settings_file = tmp_path / "settings.json"
446
+ workspace_file = tmp_path / "workspace.json"
447
+ settings_file.write_text(
448
+ json.dumps(
449
+ {
450
+ "event_log": {"timestamp_format": "absolute"},
451
+ "model": {"active_provider_id": "", "active_model": ""},
452
+ "providers": [],
453
+ "roles": [],
454
+ }
455
+ ),
456
+ encoding="utf-8",
457
+ )
458
+ workspace_file.write_text(
459
+ json.dumps(
460
+ {
461
+ "tabs": [
462
+ {
463
+ "id": "tab-1",
464
+ "title": "Restore",
465
+ "created_at": 1,
466
+ "updated_at": 1,
467
+ }
468
+ ],
469
+ "nodes": [
470
+ {
471
+ "id": "node-1",
472
+ "config": {
473
+ "node_type": "agent",
474
+ "role_name": "Worker",
475
+ "tab_id": "tab-1",
476
+ "name": "Restored Worker",
477
+ "tools": [],
478
+ "write_dirs": [],
479
+ "allow_network": False,
480
+ },
481
+ "state": "idle",
482
+ "todos": [],
483
+ "history": [],
484
+ "position": None,
485
+ "created_at": 1,
486
+ "updated_at": 1,
487
+ }
488
+ ],
489
+ "edges": [],
490
+ }
491
+ ),
492
+ encoding="utf-8",
493
+ )
494
+
495
+ monkeypatch.setattr(Agent, "start", lambda self: None)
496
+ monkeypatch.setattr(settings_module, "_SETTINGS_FILE", settings_file)
497
+ monkeypatch.setattr(settings_module, "_cached_settings", None)
498
+
499
+ bootstrap_runtime()
500
+
501
+ try:
502
+ restored = registry.get("node-1")
503
+ assert restored is not None
504
+ assert restored.state == AgentState.IDLE
505
+ assert any(
506
+ isinstance(entry, StateEntry) and entry.state == "idle"
507
+ for entry in restored.history
508
+ )
509
+ finally:
510
+ registry.reset()
511
+
512
+
513
+ @pytest.mark.parametrize("restored_state", ["running", "sleeping"])
514
+ def test_bootstrap_runtime_restores_active_nodes_as_idle(
515
+ monkeypatch,
516
+ tmp_path,
517
+ restored_state,
518
+ ):
519
+ settings_file = tmp_path / "settings.json"
520
+ workspace_file = tmp_path / "workspace.json"
521
+ registry.reset()
522
+ settings_file.write_text(
523
+ json.dumps(
524
+ {
525
+ "event_log": {"timestamp_format": "absolute"},
526
+ "model": {"active_provider_id": "", "active_model": ""},
527
+ "providers": [],
528
+ "roles": [],
529
+ }
530
+ ),
531
+ encoding="utf-8",
532
+ )
533
+ workspace_file.write_text(
534
+ json.dumps(
535
+ {
536
+ "tabs": [
537
+ {
538
+ "id": "tab-1",
539
+ "title": "Restore",
540
+ "created_at": 1,
541
+ "updated_at": 1,
542
+ }
543
+ ],
544
+ "nodes": [
545
+ {
546
+ "id": "node-1",
547
+ "config": {
548
+ "node_type": "agent",
549
+ "role_name": "Worker",
550
+ "tab_id": "tab-1",
551
+ "name": "Restored Worker",
552
+ "tools": [],
553
+ "write_dirs": [],
554
+ "allow_network": False,
555
+ },
556
+ "state": restored_state,
557
+ "todos": [{"text": "resume me"}],
558
+ "history": [
559
+ {
560
+ "type": "StateEntry",
561
+ "state": restored_state,
562
+ "reason": "before restart",
563
+ }
564
+ ],
565
+ "position": None,
566
+ "created_at": 1,
567
+ "updated_at": 1,
568
+ }
569
+ ],
570
+ "edges": [],
571
+ }
572
+ ),
573
+ encoding="utf-8",
574
+ )
575
+ monkeypatch.setattr(settings_module, "_SETTINGS_FILE", settings_file)
576
+ monkeypatch.setattr(settings_module, "_cached_settings", None)
577
+
578
+ bootstrap_runtime()
579
+
580
+ try:
581
+ restored = registry.get("node-1")
582
+ assert restored is not None
583
+ assert restored.wait_until_idle(timeout=1.0) is True
584
+ assert restored.state == AgentState.IDLE
585
+ assert restored.uuid == "node-1"
586
+ assert [todo.text for todo in restored.todos] == ["resume me"]
587
+ assert any(
588
+ isinstance(entry, StateEntry)
589
+ and entry.state == "idle"
590
+ and entry.reason == "restored"
591
+ for entry in restored.history
592
+ )
593
+ persisted = workspace_store.get_node_record("node-1")
594
+ assert persisted is not None
595
+ assert persisted.state == AgentState.IDLE
596
+ finally:
597
+ _stop_all_agents()
598
+
599
+
600
+ def test_bootstrap_runtime_restores_workflow_definition_without_runtime_connections(
601
+ monkeypatch,
602
+ tmp_path,
603
+ ):
604
+ settings_file = tmp_path / "settings.json"
605
+ workspace_file = tmp_path / "workspace.json"
606
+ registry.reset()
607
+ settings_file.write_text(
608
+ json.dumps(
609
+ {
610
+ "event_log": {"timestamp_format": "absolute"},
611
+ "model": {"active_provider_id": "", "active_model": ""},
612
+ "providers": [],
613
+ "roles": [],
614
+ }
615
+ ),
616
+ encoding="utf-8",
617
+ )
618
+ workspace_file.write_text(
619
+ json.dumps(
620
+ {
621
+ "tabs": [
622
+ {
623
+ "id": "tab-1",
624
+ "title": "Restore",
625
+ "leader_id": "leader-1",
626
+ "created_at": 1,
627
+ "updated_at": 1,
628
+ }
629
+ ],
630
+ "nodes": [
631
+ {
632
+ "id": "leader-1",
633
+ "config": {
634
+ "node_type": "agent",
635
+ "role_name": "Conductor",
636
+ "tab_id": "tab-1",
637
+ "name": "Leader",
638
+ "tools": [],
639
+ "write_dirs": [],
640
+ "allow_network": False,
641
+ },
642
+ "state": "idle",
643
+ "todos": [],
644
+ "history": [],
645
+ "position": None,
646
+ "created_at": 1,
647
+ "updated_at": 1,
648
+ },
649
+ {
650
+ "id": "node-a",
651
+ "config": {
652
+ "node_type": "agent",
653
+ "role_name": "Worker",
654
+ "tab_id": "tab-1",
655
+ "name": "Worker A",
656
+ "tools": [],
657
+ "write_dirs": [],
658
+ "allow_network": False,
659
+ },
660
+ "state": "idle",
661
+ "todos": [],
662
+ "history": [],
663
+ "position": None,
664
+ "created_at": 1,
665
+ "updated_at": 1,
666
+ },
667
+ {
668
+ "id": "node-b",
669
+ "config": {
670
+ "node_type": "agent",
671
+ "role_name": "Worker",
672
+ "tab_id": "tab-1",
673
+ "name": "Worker B",
674
+ "tools": [],
675
+ "write_dirs": [],
676
+ "allow_network": False,
677
+ },
678
+ "state": "idle",
679
+ "todos": [],
680
+ "history": [],
681
+ "position": None,
682
+ "created_at": 1,
683
+ "updated_at": 1,
684
+ },
685
+ ],
686
+ "edges": [
687
+ {
688
+ "id": "edge-1",
689
+ "tab_id": "tab-1",
690
+ "from_node_id": "leader-1",
691
+ "to_node_id": "node-a",
692
+ },
693
+ {
694
+ "id": "edge-2",
695
+ "tab_id": "tab-1",
696
+ "from_node_id": "node-a",
697
+ "to_node_id": "node-b",
698
+ },
699
+ {
700
+ "id": "edge-3",
701
+ "tab_id": "tab-1",
702
+ "from_node_id": "node-b",
703
+ "to_node_id": "node-a",
704
+ },
705
+ ],
706
+ }
707
+ ),
708
+ encoding="utf-8",
709
+ )
710
+ monkeypatch.setattr(Agent, "start", lambda self: None)
711
+ monkeypatch.setattr(settings_module, "_SETTINGS_FILE", settings_file)
712
+ monkeypatch.setattr(settings_module, "_cached_settings", None)
713
+
714
+ bootstrap_runtime()
715
+
716
+ try:
717
+ leader = registry.get("leader-1")
718
+ left = registry.get("node-a")
719
+ right = registry.get("node-b")
720
+ restored_tab = workspace_store.get_tab("tab-1")
721
+
722
+ assert leader is not None
723
+ assert left is not None
724
+ assert right is not None
725
+ assert restored_tab is not None
726
+ assert leader.get_connections_snapshot() == []
727
+ assert left.get_connections_snapshot() == []
728
+ assert right.get_connections_snapshot() == []
729
+ assert {node.id for node in restored_tab.definition.nodes} == {
730
+ "node-a",
731
+ "node-b",
732
+ }
733
+ assert {
734
+ (edge.from_node_id, edge.to_node_id)
735
+ for edge in restored_tab.definition.edges
736
+ } == {
737
+ ("node-a", "node-b"),
738
+ ("node-b", "node-a"),
739
+ }
740
+ finally:
741
+ registry.reset()
742
+
743
+
744
+ def test_bootstrap_runtime_preserves_error_state_for_restored_nodes(
745
+ monkeypatch,
746
+ tmp_path,
747
+ ):
748
+ settings_file = tmp_path / "settings.json"
749
+ workspace_file = tmp_path / "workspace.json"
750
+ registry.reset()
751
+ settings_file.write_text(
752
+ json.dumps(
753
+ {
754
+ "event_log": {"timestamp_format": "absolute"},
755
+ "model": {"active_provider_id": "", "active_model": ""},
756
+ "providers": [],
757
+ "roles": [],
758
+ }
759
+ ),
760
+ encoding="utf-8",
761
+ )
762
+ workspace_file.write_text(
763
+ json.dumps(
764
+ {
765
+ "tabs": [
766
+ {
767
+ "id": "tab-1",
768
+ "title": "Restore",
769
+ "created_at": 1,
770
+ "updated_at": 1,
771
+ }
772
+ ],
773
+ "nodes": [
774
+ {
775
+ "id": "node-1",
776
+ "config": {
777
+ "node_type": "agent",
778
+ "role_name": "Worker",
779
+ "tab_id": "tab-1",
780
+ "name": "Broken Worker",
781
+ "tools": [],
782
+ "write_dirs": [],
783
+ "allow_network": False,
784
+ },
785
+ "state": "error",
786
+ "todos": [],
787
+ "history": [
788
+ {
789
+ "type": "StateEntry",
790
+ "state": "error",
791
+ "reason": "provider failure",
792
+ }
793
+ ],
794
+ "position": None,
795
+ "created_at": 1,
796
+ "updated_at": 1,
797
+ }
798
+ ],
799
+ "edges": [],
800
+ }
801
+ ),
802
+ encoding="utf-8",
803
+ )
804
+ monkeypatch.setattr(settings_module, "_SETTINGS_FILE", settings_file)
805
+ monkeypatch.setattr(settings_module, "_cached_settings", None)
806
+
807
+ bootstrap_runtime()
808
+
809
+ try:
810
+ restored = registry.get("node-1")
811
+ assert restored is not None
812
+ assert restored.state == AgentState.ERROR
813
+ assert restored.wait_until_idle(timeout=0.05) is False
814
+ assert not any(
815
+ isinstance(entry, StateEntry)
816
+ and entry.state == "idle"
817
+ and entry.reason == "initialized, awaiting first message"
818
+ for entry in restored.history
819
+ )
820
+ persisted = workspace_store.get_node_record("node-1")
821
+ assert persisted is not None
822
+ assert persisted.state == AgentState.ERROR
823
+ finally:
824
+ _stop_all_agents()
825
+
826
+
827
+ def test_bootstrap_runtime_skips_terminated_restored_nodes(monkeypatch, tmp_path):
828
+ settings_file = tmp_path / "settings.json"
829
+ workspace_file = tmp_path / "workspace.json"
830
+ registry.reset()
831
+ settings_file.write_text(
832
+ json.dumps(
833
+ {
834
+ "event_log": {"timestamp_format": "absolute"},
835
+ "model": {"active_provider_id": "", "active_model": ""},
836
+ "providers": [],
837
+ "roles": [],
838
+ }
839
+ ),
840
+ encoding="utf-8",
841
+ )
842
+ workspace_file.write_text(
843
+ json.dumps(
844
+ {
845
+ "tabs": [
846
+ {
847
+ "id": "tab-1",
848
+ "title": "Restore",
849
+ "created_at": 1,
850
+ "updated_at": 1,
851
+ }
852
+ ],
853
+ "nodes": [
854
+ {
855
+ "id": "node-1",
856
+ "config": {
857
+ "node_type": "agent",
858
+ "role_name": "Worker",
859
+ "tab_id": "tab-1",
860
+ "name": "Finished Worker",
861
+ "tools": [],
862
+ "write_dirs": [],
863
+ "allow_network": False,
864
+ },
865
+ "state": "terminated",
866
+ "todos": [],
867
+ "history": [
868
+ {
869
+ "type": "StateEntry",
870
+ "state": "terminated",
871
+ "reason": "done",
872
+ }
873
+ ],
874
+ "position": None,
875
+ "created_at": 1,
876
+ "updated_at": 1,
877
+ }
878
+ ],
879
+ "edges": [],
880
+ }
881
+ ),
882
+ encoding="utf-8",
883
+ )
884
+ monkeypatch.setattr(settings_module, "_SETTINGS_FILE", settings_file)
885
+ monkeypatch.setattr(settings_module, "_cached_settings", None)
886
+
887
+ bootstrap_runtime()
888
+
889
+ try:
890
+ assert registry.get("node-1") is None
891
+ persisted = workspace_store.get_node_record("node-1")
892
+ assert persisted is not None
893
+ assert persisted.state == AgentState.TERMINATED
894
+ finally:
895
+ _stop_all_agents()
896
+
897
+
898
+ def test_shutdown_runtime_keeps_persistent_workspace_nodes_unterminated(
899
+ monkeypatch,
900
+ tmp_path,
901
+ ):
902
+ settings_file = tmp_path / "settings.json"
903
+ workspace_file = tmp_path / "workspace.json"
904
+ registry.reset()
905
+ settings_file.write_text(
906
+ json.dumps(
907
+ {
908
+ "event_log": {"timestamp_format": "absolute"},
909
+ "model": {"active_provider_id": "", "active_model": ""},
910
+ "providers": [],
911
+ "roles": [],
912
+ }
913
+ ),
914
+ encoding="utf-8",
915
+ )
916
+ workspace_file.write_text(
917
+ json.dumps(
918
+ {
919
+ "tabs": [
920
+ {
921
+ "id": "tab-1",
922
+ "title": "Restore",
923
+ "created_at": 1,
924
+ "updated_at": 1,
925
+ }
926
+ ],
927
+ "nodes": [
928
+ {
929
+ "id": "node-1",
930
+ "config": {
931
+ "node_type": "agent",
932
+ "role_name": "Worker",
933
+ "tab_id": "tab-1",
934
+ "name": "Persistent Worker",
935
+ "tools": [],
936
+ "write_dirs": [],
937
+ "allow_network": False,
938
+ },
939
+ "state": "idle",
940
+ "todos": [],
941
+ "history": [
942
+ {
943
+ "type": "StateEntry",
944
+ "state": "idle",
945
+ "reason": "restored",
946
+ }
947
+ ],
948
+ "position": None,
949
+ "created_at": 1,
950
+ "updated_at": 1,
951
+ }
952
+ ],
953
+ "edges": [],
954
+ }
955
+ ),
956
+ encoding="utf-8",
957
+ )
958
+ monkeypatch.setattr(settings_module, "_SETTINGS_FILE", settings_file)
959
+ monkeypatch.setattr(settings_module, "_cached_settings", None)
960
+
961
+ bootstrap_runtime()
962
+
963
+ try:
964
+ assistant = registry.get_assistant()
965
+ persistent = registry.get("node-1")
966
+ assert assistant is not None
967
+ assert persistent is not None
968
+ assistant_id = assistant.uuid
969
+ assistant._append_history(ReceivedMessage(content="hello", from_id="human"))
970
+ assistant._append_history(AssistantText(content="hi there"))
971
+
972
+ shutdown_runtime()
973
+
974
+ assert assistant.wait_for_termination(timeout=1.0) is True
975
+ assert persistent.wait_for_termination(timeout=1.0) is True
976
+ assert registry.get_all() == []
977
+ assert assistant.state == AgentState.IDLE
978
+ assert persistent.state == AgentState.IDLE
979
+ persisted_assistant = workspace_store.get_node_record(assistant_id)
980
+ assert persisted_assistant is not None
981
+ assert persisted_assistant.state == AgentState.IDLE
982
+ assert any(
983
+ isinstance(entry, ReceivedMessage) and entry.content == "hello"
984
+ for entry in persisted_assistant.history
985
+ )
986
+ assert any(
987
+ isinstance(entry, AssistantText) and entry.content == "hi there"
988
+ for entry in persisted_assistant.history
989
+ )
990
+ persisted = workspace_store.get_node_record("node-1")
991
+ assert persisted is not None
992
+ assert persisted.state == AgentState.IDLE
993
+ assert not any(
994
+ isinstance(entry, StateEntry) and entry.state == "terminated"
995
+ for entry in persistent.history
996
+ )
997
+
998
+ bootstrap_runtime()
999
+
1000
+ restored_assistant = registry.get_assistant()
1001
+ assert restored_assistant is not None
1002
+ assert restored_assistant.uuid == assistant_id
1003
+ assert any(
1004
+ isinstance(entry, ReceivedMessage) and entry.content == "hello"
1005
+ for entry in restored_assistant.history
1006
+ )
1007
+ assert any(
1008
+ isinstance(entry, AssistantText) and entry.content == "hi there"
1009
+ for entry in restored_assistant.history
1010
+ )
1011
+ finally:
1012
+ _stop_all_agents()