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,711 @@
1
+ import json
2
+ from dataclasses import asdict
3
+
4
+ import flowent.settings as settings_module
5
+ from flowent.access import set_access_code, verify_access_code
6
+ from flowent.settings import (
7
+ CONDUCTOR_ROLE_DESCRIPTION,
8
+ CONDUCTOR_ROLE_INCLUDED_TOOLS,
9
+ CONDUCTOR_ROLE_NAME,
10
+ CONDUCTOR_ROLE_SYSTEM_PROMPT,
11
+ DESIGNER_ROLE_DESCRIPTION,
12
+ DESIGNER_ROLE_INCLUDED_TOOLS,
13
+ DESIGNER_ROLE_NAME,
14
+ DESIGNER_ROLE_SYSTEM_PROMPT,
15
+ STEWARD_ROLE_DESCRIPTION,
16
+ STEWARD_ROLE_INCLUDED_TOOLS,
17
+ STEWARD_ROLE_NAME,
18
+ STEWARD_ROLE_SYSTEM_PROMPT,
19
+ WORKER_ROLE_DESCRIPTION,
20
+ WORKER_ROLE_INCLUDED_TOOLS,
21
+ WORKER_ROLE_NAME,
22
+ WORKER_ROLE_SYSTEM_PROMPT,
23
+ AssistantSettings,
24
+ LeaderSettings,
25
+ ProviderConfig,
26
+ RoleConfig,
27
+ RoleModelConfig,
28
+ Settings,
29
+ TelegramApprovedChat,
30
+ TelegramSettings,
31
+ build_default_assistant_write_dirs,
32
+ )
33
+
34
+
35
+ def test_load_settings_migrates_legacy_role_field(monkeypatch, tmp_path):
36
+ settings_file = tmp_path / "settings.json"
37
+ settings_file.write_text(
38
+ json.dumps(
39
+ {
40
+ "event_log": {"timestamp_format": "absolute"},
41
+ "model": {
42
+ "active_provider_id": "provider-1",
43
+ "active_model": "gpt-default",
44
+ },
45
+ "providers": [],
46
+ "roles": [
47
+ {
48
+ "id": "legacy-worker",
49
+ "name": "Worker",
50
+ "system_prompt": "Do work.",
51
+ "required_tools": ["read"],
52
+ }
53
+ ],
54
+ }
55
+ ),
56
+ encoding="utf-8",
57
+ )
58
+
59
+ monkeypatch.setattr(settings_module, "_SETTINGS_FILE", settings_file)
60
+ monkeypatch.setattr(settings_module, "_cached_settings", None)
61
+
62
+ loaded = settings_module.load_settings()
63
+
64
+ assert loaded.roles == [
65
+ RoleConfig(
66
+ name="Worker",
67
+ description="Do work.",
68
+ system_prompt="Do work.",
69
+ included_tools=["read"],
70
+ )
71
+ ]
72
+ assert loaded.assistant == AssistantSettings()
73
+ assert loaded.leader == LeaderSettings()
74
+ assert loaded.telegram == TelegramSettings()
75
+
76
+ persisted = json.loads(settings_file.read_text(encoding="utf-8"))
77
+ assert persisted["assistant"] == {
78
+ "role_name": STEWARD_ROLE_NAME,
79
+ "allow_network": True,
80
+ "write_dirs": build_default_assistant_write_dirs(),
81
+ }
82
+ assert persisted["leader"] == {"role_name": CONDUCTOR_ROLE_NAME}
83
+ assert persisted["telegram"] == {
84
+ "bot_token": "",
85
+ "pending_chats": [],
86
+ "approved_chats": [],
87
+ }
88
+ assert persisted["roles"] == [
89
+ {
90
+ "name": "Worker",
91
+ "description": "Do work.",
92
+ "system_prompt": "Do work.",
93
+ "model": None,
94
+ "model_params": None,
95
+ "included_tools": ["read"],
96
+ "excluded_tools": [],
97
+ }
98
+ ]
99
+
100
+
101
+ def test_load_settings_defaults_model_max_retries(monkeypatch, tmp_path):
102
+ settings_file = tmp_path / "settings.json"
103
+ settings_file.write_text(
104
+ json.dumps(
105
+ {
106
+ "event_log": {"timestamp_format": "absolute"},
107
+ "model": {"active_provider_id": "", "active_model": ""},
108
+ "providers": [],
109
+ "roles": [],
110
+ }
111
+ ),
112
+ encoding="utf-8",
113
+ )
114
+
115
+ monkeypatch.setattr(settings_module, "_SETTINGS_FILE", settings_file)
116
+ monkeypatch.setattr(settings_module, "_cached_settings", None)
117
+
118
+ loaded = settings_module.load_settings()
119
+
120
+ assert loaded.model.max_retries == 5
121
+
122
+ persisted = json.loads(settings_file.read_text(encoding="utf-8"))
123
+ assert persisted["model"]["max_retries"] == 5
124
+
125
+
126
+ def test_load_settings_defaults_assistant_permissions(monkeypatch, tmp_path):
127
+ settings_file = tmp_path / "settings.json"
128
+ settings_file.write_text(
129
+ json.dumps(
130
+ {
131
+ "event_log": {"timestamp_format": "absolute"},
132
+ "model": {"active_provider_id": "", "active_model": ""},
133
+ "providers": [],
134
+ "roles": [],
135
+ }
136
+ ),
137
+ encoding="utf-8",
138
+ )
139
+
140
+ monkeypatch.setattr(settings_module, "_SETTINGS_FILE", settings_file)
141
+ monkeypatch.setattr(settings_module, "_cached_settings", None)
142
+
143
+ loaded = settings_module.load_settings()
144
+
145
+ assert loaded.assistant.allow_network is True
146
+ assert loaded.assistant.write_dirs == build_default_assistant_write_dirs()
147
+
148
+ persisted = json.loads(settings_file.read_text(encoding="utf-8"))
149
+ assert persisted["assistant"] == {
150
+ "role_name": STEWARD_ROLE_NAME,
151
+ "allow_network": True,
152
+ "write_dirs": build_default_assistant_write_dirs(),
153
+ }
154
+
155
+
156
+ def test_load_settings_normalizes_assistant_write_dirs(monkeypatch, tmp_path):
157
+ settings_file = tmp_path / "settings.json"
158
+ settings_file.write_text(
159
+ json.dumps(
160
+ {
161
+ "assistant": {
162
+ "role_name": STEWARD_ROLE_NAME,
163
+ "allow_network": False,
164
+ "write_dirs": [" ./workspace ", "./workspace/", ""],
165
+ },
166
+ "event_log": {"timestamp_format": "absolute"},
167
+ "model": {"active_provider_id": "", "active_model": ""},
168
+ "providers": [],
169
+ "roles": [],
170
+ }
171
+ ),
172
+ encoding="utf-8",
173
+ )
174
+
175
+ monkeypatch.setattr(settings_module, "_SETTINGS_FILE", settings_file)
176
+ monkeypatch.setattr(settings_module, "_cached_settings", None)
177
+
178
+ loaded = settings_module.load_settings()
179
+ expected_write_dirs = [str((settings_module.WORKING_DIR / "workspace").resolve())]
180
+
181
+ assert loaded.assistant == AssistantSettings(
182
+ role_name=STEWARD_ROLE_NAME,
183
+ allow_network=False,
184
+ write_dirs=expected_write_dirs,
185
+ )
186
+
187
+ persisted = json.loads(settings_file.read_text(encoding="utf-8"))
188
+ assert persisted["assistant"] == {
189
+ "role_name": STEWARD_ROLE_NAME,
190
+ "allow_network": False,
191
+ "write_dirs": expected_write_dirs,
192
+ }
193
+
194
+
195
+ def test_load_settings_defaults_model_retry_policy(monkeypatch, tmp_path):
196
+ settings_file = tmp_path / "settings.json"
197
+ settings_file.write_text(
198
+ json.dumps(
199
+ {
200
+ "event_log": {"timestamp_format": "absolute"},
201
+ "model": {"active_provider_id": "", "active_model": ""},
202
+ "providers": [],
203
+ "roles": [],
204
+ }
205
+ ),
206
+ encoding="utf-8",
207
+ )
208
+
209
+ monkeypatch.setattr(settings_module, "_SETTINGS_FILE", settings_file)
210
+ monkeypatch.setattr(settings_module, "_cached_settings", None)
211
+
212
+ loaded = settings_module.load_settings()
213
+
214
+ assert loaded.model.retry_policy == "limited"
215
+
216
+ persisted = json.loads(settings_file.read_text(encoding="utf-8"))
217
+ assert persisted["model"]["retry_policy"] == "limited"
218
+
219
+
220
+ def test_load_settings_defaults_retry_backoff_fields(monkeypatch, tmp_path):
221
+ settings_file = tmp_path / "settings.json"
222
+ settings_file.write_text(
223
+ json.dumps(
224
+ {
225
+ "event_log": {"timestamp_format": "absolute"},
226
+ "model": {"active_provider_id": "", "active_model": ""},
227
+ "providers": [],
228
+ "roles": [],
229
+ }
230
+ ),
231
+ encoding="utf-8",
232
+ )
233
+
234
+ monkeypatch.setattr(settings_module, "_SETTINGS_FILE", settings_file)
235
+ monkeypatch.setattr(settings_module, "_cached_settings", None)
236
+
237
+ loaded = settings_module.load_settings()
238
+
239
+ assert loaded.model.retry_initial_delay_seconds == 0.5
240
+ assert loaded.model.retry_max_delay_seconds == 8.0
241
+ assert loaded.model.retry_backoff_cap_retries == 5
242
+
243
+ persisted = json.loads(settings_file.read_text(encoding="utf-8"))
244
+ assert persisted["model"]["retry_initial_delay_seconds"] == 0.5
245
+ assert persisted["model"]["retry_max_delay_seconds"] == 8.0
246
+ assert persisted["model"]["retry_backoff_cap_retries"] == 5
247
+
248
+
249
+ def test_get_settings_reloads_when_settings_file_changes_externally(
250
+ monkeypatch, tmp_path
251
+ ):
252
+ settings_file = tmp_path / "settings.json"
253
+
254
+ monkeypatch.setattr(settings_module, "_SETTINGS_FILE", settings_file)
255
+ monkeypatch.setattr(settings_module, "_cached_settings", None)
256
+ monkeypatch.setattr(settings_module, "_cached_settings_file_signature", None)
257
+
258
+ settings_module.save_settings(Settings(custom_prompt="before"))
259
+
260
+ cached = settings_module.get_settings()
261
+ assert cached.custom_prompt == "before"
262
+
263
+ persisted = json.loads(settings_file.read_text(encoding="utf-8"))
264
+ persisted["custom_prompt"] = "after external update"
265
+ settings_file.write_text(json.dumps(persisted, indent=2), encoding="utf-8")
266
+
267
+ reloaded = settings_module.get_settings()
268
+
269
+ assert reloaded.custom_prompt == "after external update"
270
+
271
+
272
+ def test_save_settings_preserves_newer_live_access_when_cache_is_stale(
273
+ monkeypatch, tmp_path
274
+ ):
275
+ settings_file = tmp_path / "settings.json"
276
+
277
+ monkeypatch.setattr(settings_module, "_SETTINGS_FILE", settings_file)
278
+ monkeypatch.setattr(settings_module, "_cached_settings", None)
279
+ monkeypatch.setattr(settings_module, "_cached_settings_file_signature", None)
280
+
281
+ initial = Settings()
282
+ set_access_code(initial, "OLD-CODE")
283
+ settings_module.save_settings(initial)
284
+
285
+ cached = settings_module.get_settings()
286
+ assert verify_access_code(cached.access, "OLD-CODE")
287
+
288
+ refreshed, _ = settings_module._read_settings_file()
289
+ set_access_code(refreshed, "NEW-CODE")
290
+ settings_file.write_text(
291
+ json.dumps(asdict(refreshed), indent=2),
292
+ encoding="utf-8",
293
+ )
294
+
295
+ cached.providers.append(
296
+ ProviderConfig(
297
+ id="provider-1",
298
+ name="Primary",
299
+ type="openai_compatible",
300
+ base_url="https://api.example.com/v1",
301
+ api_key="secret",
302
+ )
303
+ )
304
+ settings_module.save_settings(cached)
305
+
306
+ final_settings, _ = settings_module._read_settings_file()
307
+
308
+ assert len(final_settings.providers) == 1
309
+ assert final_settings.providers[0].id == "provider-1"
310
+ assert verify_access_code(final_settings.access, "NEW-CODE")
311
+ assert not verify_access_code(final_settings.access, "OLD-CODE")
312
+
313
+
314
+ def test_load_settings_defaults_model_timeout_ms(monkeypatch, tmp_path):
315
+ settings_file = tmp_path / "settings.json"
316
+ settings_file.write_text(
317
+ json.dumps(
318
+ {
319
+ "event_log": {"timestamp_format": "absolute"},
320
+ "model": {"active_provider_id": "", "active_model": ""},
321
+ "providers": [],
322
+ "roles": [],
323
+ }
324
+ ),
325
+ encoding="utf-8",
326
+ )
327
+
328
+ monkeypatch.setattr(settings_module, "_SETTINGS_FILE", settings_file)
329
+ monkeypatch.setattr(settings_module, "_cached_settings", None)
330
+
331
+ loaded = settings_module.load_settings()
332
+
333
+ assert loaded.model.timeout_ms == 10000
334
+
335
+ persisted = json.loads(settings_file.read_text(encoding="utf-8"))
336
+ assert persisted["model"]["timeout_ms"] == 10000
337
+
338
+
339
+ def test_load_settings_normalizes_provider_headers(monkeypatch, tmp_path):
340
+ settings_file = tmp_path / "settings.json"
341
+ settings_file.write_text(
342
+ json.dumps(
343
+ {
344
+ "event_log": {"timestamp_format": "absolute"},
345
+ "model": {"active_provider_id": "", "active_model": ""},
346
+ "providers": [
347
+ {
348
+ "id": "provider-1",
349
+ "name": "Primary",
350
+ "type": "openai_compatible",
351
+ "base_url": "https://api.example.com/v1",
352
+ "api_key": "secret",
353
+ "headers": {
354
+ "Authorization": "Bearer test",
355
+ "X-Number": 1,
356
+ },
357
+ }
358
+ ],
359
+ "roles": [],
360
+ }
361
+ ),
362
+ encoding="utf-8",
363
+ )
364
+
365
+ monkeypatch.setattr(settings_module, "_SETTINGS_FILE", settings_file)
366
+ monkeypatch.setattr(settings_module, "_cached_settings", None)
367
+
368
+ loaded = settings_module.load_settings()
369
+
370
+ assert loaded.providers == [
371
+ ProviderConfig(
372
+ id="provider-1",
373
+ name="Primary",
374
+ type="openai_compatible",
375
+ base_url="https://api.example.com/v1",
376
+ api_key="secret",
377
+ headers={"Authorization": "Bearer test"},
378
+ )
379
+ ]
380
+
381
+ persisted = json.loads(settings_file.read_text(encoding="utf-8"))
382
+ assert persisted["providers"][0]["headers"] == {"Authorization": "Bearer test"}
383
+
384
+
385
+ def test_load_settings_defaults_provider_retry_429_delay_seconds(monkeypatch, tmp_path):
386
+ settings_file = tmp_path / "settings.json"
387
+ settings_file.write_text(
388
+ json.dumps(
389
+ {
390
+ "event_log": {"timestamp_format": "absolute"},
391
+ "model": {"active_provider_id": "", "active_model": ""},
392
+ "providers": [
393
+ {
394
+ "id": "provider-1",
395
+ "name": "Primary",
396
+ "type": "openai_compatible",
397
+ "base_url": "https://api.example.com/v1",
398
+ "api_key": "secret",
399
+ }
400
+ ],
401
+ "roles": [],
402
+ }
403
+ ),
404
+ encoding="utf-8",
405
+ )
406
+
407
+ monkeypatch.setattr(settings_module, "_SETTINGS_FILE", settings_file)
408
+ monkeypatch.setattr(settings_module, "_cached_settings", None)
409
+
410
+ loaded = settings_module.load_settings()
411
+
412
+ assert loaded.providers[0].retry_429_delay_seconds == 0
413
+
414
+ persisted = json.loads(settings_file.read_text(encoding="utf-8"))
415
+ assert persisted["providers"][0]["retry_429_delay_seconds"] == 0
416
+
417
+
418
+ def test_load_settings_drops_removed_exit_tool_from_roles(monkeypatch, tmp_path):
419
+ settings_file = tmp_path / "settings.json"
420
+ settings_file.write_text(
421
+ json.dumps(
422
+ {
423
+ "event_log": {"timestamp_format": "absolute"},
424
+ "model": {"active_provider_id": "", "active_model": ""},
425
+ "providers": [],
426
+ "roles": [
427
+ {
428
+ "name": "Worker",
429
+ "system_prompt": "Do work.",
430
+ "included_tools": ["read", "exit", "exec"],
431
+ "excluded_tools": ["exit", "fetch"],
432
+ }
433
+ ],
434
+ }
435
+ ),
436
+ encoding="utf-8",
437
+ )
438
+
439
+ monkeypatch.setattr(settings_module, "_SETTINGS_FILE", settings_file)
440
+ monkeypatch.setattr(settings_module, "_cached_settings", None)
441
+
442
+ loaded = settings_module.load_settings()
443
+
444
+ assert loaded.roles == [
445
+ RoleConfig(
446
+ name="Worker",
447
+ description="Do work.",
448
+ system_prompt="Do work.",
449
+ included_tools=["read", "exec"],
450
+ excluded_tools=["fetch"],
451
+ )
452
+ ]
453
+
454
+ persisted = json.loads(settings_file.read_text(encoding="utf-8"))
455
+ assert persisted["roles"] == [
456
+ {
457
+ "name": "Worker",
458
+ "description": "Do work.",
459
+ "system_prompt": "Do work.",
460
+ "model": None,
461
+ "model_params": None,
462
+ "included_tools": ["read", "exec"],
463
+ "excluded_tools": ["fetch"],
464
+ }
465
+ ]
466
+
467
+
468
+ def test_load_settings_migrates_legacy_post_prompt_to_custom_post_prompt(
469
+ monkeypatch, tmp_path
470
+ ):
471
+ settings_file = tmp_path / "settings.json"
472
+ settings_file.write_text(
473
+ json.dumps(
474
+ {
475
+ "event_log": {"timestamp_format": "absolute"},
476
+ "model": {"active_provider_id": "", "active_model": ""},
477
+ "custom_prompt": "Apply extra guardrails.",
478
+ "post_prompt": "Append runtime guidance.",
479
+ "providers": [],
480
+ "roles": [],
481
+ }
482
+ ),
483
+ encoding="utf-8",
484
+ )
485
+
486
+ monkeypatch.setattr(settings_module, "_SETTINGS_FILE", settings_file)
487
+ monkeypatch.setattr(settings_module, "_cached_settings", None)
488
+
489
+ loaded = settings_module.load_settings()
490
+
491
+ assert loaded.custom_prompt == "Apply extra guardrails."
492
+ assert loaded.custom_post_prompt == "Append runtime guidance."
493
+ assert loaded.assistant == AssistantSettings()
494
+ assert loaded.leader == LeaderSettings()
495
+ assert loaded.telegram == TelegramSettings()
496
+
497
+ persisted = json.loads(settings_file.read_text(encoding="utf-8"))
498
+ assert persisted["custom_post_prompt"] == "Append runtime guidance."
499
+ assert "post_prompt" not in persisted
500
+
501
+
502
+ def test_load_settings_migrates_legacy_model_override(monkeypatch, tmp_path):
503
+ settings_file = tmp_path / "settings.json"
504
+ settings_file.write_text(
505
+ json.dumps(
506
+ {
507
+ "event_log": {"timestamp_format": "absolute"},
508
+ "model": {
509
+ "active_provider_id": "provider-1",
510
+ "active_model": "gpt-default",
511
+ },
512
+ "providers": [],
513
+ "roles": [
514
+ {
515
+ "name": "Reviewer",
516
+ "system_prompt": "Review carefully.",
517
+ "model_override": "gpt-4.1-mini",
518
+ "included_tools": ["read", "exec"],
519
+ "excluded_tools": ["fetch"],
520
+ }
521
+ ],
522
+ }
523
+ ),
524
+ encoding="utf-8",
525
+ )
526
+
527
+ monkeypatch.setattr(settings_module, "_SETTINGS_FILE", settings_file)
528
+ monkeypatch.setattr(settings_module, "_cached_settings", None)
529
+
530
+ loaded = settings_module.load_settings()
531
+
532
+ assert loaded.roles == [
533
+ RoleConfig(
534
+ name="Reviewer",
535
+ description="Review carefully.",
536
+ system_prompt="Review carefully.",
537
+ model=RoleModelConfig(
538
+ provider_id="provider-1",
539
+ model="gpt-4.1-mini",
540
+ ),
541
+ included_tools=["read", "exec"],
542
+ excluded_tools=["fetch"],
543
+ )
544
+ ]
545
+
546
+ persisted = json.loads(settings_file.read_text(encoding="utf-8"))
547
+ assert persisted["roles"] == [
548
+ {
549
+ "name": "Reviewer",
550
+ "description": "Review carefully.",
551
+ "system_prompt": "Review carefully.",
552
+ "model": {
553
+ "provider_id": "provider-1",
554
+ "model": "gpt-4.1-mini",
555
+ },
556
+ "model_params": None,
557
+ "included_tools": ["read", "exec"],
558
+ "excluded_tools": ["fetch"],
559
+ }
560
+ ]
561
+
562
+
563
+ def test_load_settings_parses_role_model_object(monkeypatch, tmp_path):
564
+ settings_file = tmp_path / "settings.json"
565
+ settings_file.write_text(
566
+ json.dumps(
567
+ {
568
+ "event_log": {"timestamp_format": "absolute"},
569
+ "model": {
570
+ "active_provider_id": "provider-1",
571
+ "active_model": "gpt-default",
572
+ },
573
+ "providers": [],
574
+ "roles": [
575
+ {
576
+ "name": "Reviewer",
577
+ "system_prompt": "Review carefully.",
578
+ "model": {
579
+ "provider_id": "provider-2",
580
+ "model": "gpt-4.1-mini",
581
+ },
582
+ }
583
+ ],
584
+ }
585
+ ),
586
+ encoding="utf-8",
587
+ )
588
+
589
+ monkeypatch.setattr(settings_module, "_SETTINGS_FILE", settings_file)
590
+ monkeypatch.setattr(settings_module, "_cached_settings", None)
591
+
592
+ loaded = settings_module.load_settings()
593
+
594
+ assert loaded.roles == [
595
+ RoleConfig(
596
+ name="Reviewer",
597
+ description="Review carefully.",
598
+ system_prompt="Review carefully.",
599
+ model=RoleModelConfig(
600
+ provider_id="provider-2",
601
+ model="gpt-4.1-mini",
602
+ ),
603
+ )
604
+ ]
605
+
606
+
607
+ def test_load_settings_parses_telegram_settings(monkeypatch, tmp_path):
608
+ settings_file = tmp_path / "settings.json"
609
+ settings_file.write_text(
610
+ json.dumps(
611
+ {
612
+ "event_log": {"timestamp_format": "absolute"},
613
+ "assistant": {
614
+ "role_name": STEWARD_ROLE_NAME,
615
+ "allow_network": True,
616
+ "write_dirs": build_default_assistant_write_dirs(),
617
+ },
618
+ "telegram": {
619
+ "bot_token": "123456:ABCDE",
620
+ "registered_chat_ids": ["-1001", 2002],
621
+ },
622
+ "model": {"active_provider_id": "", "active_model": ""},
623
+ "providers": [],
624
+ "roles": [],
625
+ }
626
+ ),
627
+ encoding="utf-8",
628
+ )
629
+
630
+ monkeypatch.setattr(settings_module, "_SETTINGS_FILE", settings_file)
631
+ monkeypatch.setattr(settings_module, "_cached_settings", None)
632
+
633
+ loaded = settings_module.load_settings()
634
+
635
+ assert loaded.telegram == TelegramSettings(
636
+ bot_token="123456:ABCDE",
637
+ pending_chats=[],
638
+ approved_chats=[
639
+ TelegramApprovedChat(chat_id=-1001, approved_at=0.0),
640
+ TelegramApprovedChat(chat_id=2002, approved_at=0.0),
641
+ ],
642
+ )
643
+
644
+ persisted = json.loads(settings_file.read_text(encoding="utf-8"))
645
+ assert persisted["telegram"] == {
646
+ "bot_token": "123456:ABCDE",
647
+ "pending_chats": [],
648
+ "approved_chats": [
649
+ {
650
+ "chat_id": -1001,
651
+ "username": None,
652
+ "display_name": "",
653
+ "approved_at": 0.0,
654
+ },
655
+ {
656
+ "chat_id": 2002,
657
+ "username": None,
658
+ "display_name": "",
659
+ "approved_at": 0.0,
660
+ },
661
+ ],
662
+ }
663
+
664
+
665
+ def test_ensure_builtin_roles_repairs_and_creates_builtin_roles():
666
+ settings = Settings(
667
+ roles=[
668
+ RoleConfig(
669
+ name=WORKER_ROLE_NAME,
670
+ system_prompt="Outdated prompt.",
671
+ included_tools=[],
672
+ excluded_tools=["fetch"],
673
+ )
674
+ ]
675
+ )
676
+
677
+ changed = settings_module.ensure_builtin_roles(settings)
678
+
679
+ assert changed is True
680
+ assert settings.roles == [
681
+ RoleConfig(
682
+ name=STEWARD_ROLE_NAME,
683
+ description=STEWARD_ROLE_DESCRIPTION,
684
+ system_prompt=STEWARD_ROLE_SYSTEM_PROMPT,
685
+ included_tools=STEWARD_ROLE_INCLUDED_TOOLS,
686
+ excluded_tools=[],
687
+ ),
688
+ RoleConfig(
689
+ name=WORKER_ROLE_NAME,
690
+ description=WORKER_ROLE_DESCRIPTION,
691
+ system_prompt=WORKER_ROLE_SYSTEM_PROMPT,
692
+ included_tools=WORKER_ROLE_INCLUDED_TOOLS,
693
+ excluded_tools=[],
694
+ ),
695
+ RoleConfig(
696
+ name=CONDUCTOR_ROLE_NAME,
697
+ description=CONDUCTOR_ROLE_DESCRIPTION,
698
+ system_prompt=CONDUCTOR_ROLE_SYSTEM_PROMPT,
699
+ included_tools=CONDUCTOR_ROLE_INCLUDED_TOOLS,
700
+ excluded_tools=[],
701
+ ),
702
+ RoleConfig(
703
+ name=DESIGNER_ROLE_NAME,
704
+ description=DESIGNER_ROLE_DESCRIPTION,
705
+ system_prompt=DESIGNER_ROLE_SYSTEM_PROMPT,
706
+ included_tools=DESIGNER_ROLE_INCLUDED_TOOLS,
707
+ excluded_tools=[],
708
+ ),
709
+ ]
710
+ assert settings.assistant.role_name == STEWARD_ROLE_NAME
711
+ assert settings.leader.role_name == CONDUCTOR_ROLE_NAME