flowent 0.0.7 → 0.0.11

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 (387) hide show
  1. package/README.md +0 -3
  2. package/backend/README.md +0 -3
  3. package/backend/pyproject.toml +2 -8
  4. package/backend/src/flowent/__init__.py +6 -2
  5. package/backend/src/flowent/__pycache__/__init__.cpython-313.pyc +0 -0
  6. package/backend/src/flowent/__pycache__/agent.cpython-313.pyc +0 -0
  7. package/backend/src/flowent/__pycache__/context.cpython-313.pyc +0 -0
  8. package/backend/src/flowent/__pycache__/llm.cpython-313.pyc +0 -0
  9. package/backend/src/flowent/__pycache__/logging.cpython-313.pyc +0 -0
  10. package/backend/src/flowent/__pycache__/main.cpython-313.pyc +0 -0
  11. package/backend/src/flowent/__pycache__/patch.cpython-313.pyc +0 -0
  12. package/backend/src/flowent/__pycache__/paths.cpython-313.pyc +0 -0
  13. package/backend/src/flowent/__pycache__/sandbox.cpython-313.pyc +0 -0
  14. package/backend/src/flowent/__pycache__/storage.cpython-313.pyc +0 -0
  15. package/backend/src/flowent/__pycache__/tools.cpython-313.pyc +0 -0
  16. package/backend/src/flowent/agent.py +213 -3173
  17. package/backend/src/flowent/cli.py +19 -24
  18. package/backend/src/flowent/context.py +127 -0
  19. package/backend/src/flowent/llm.py +256 -0
  20. package/backend/src/flowent/logging.py +170 -129
  21. package/backend/src/flowent/main.py +321 -70
  22. package/backend/src/flowent/patch.py +182 -0
  23. package/backend/src/flowent/paths.py +11 -0
  24. package/backend/src/flowent/sandbox.py +214 -40
  25. package/backend/src/flowent/static/assets/geist-cyrillic-wght-normal-CHSlOQsW.woff2 +0 -0
  26. package/backend/src/flowent/static/assets/geist-latin-ext-wght-normal-DMtmJ5ZE.woff2 +0 -0
  27. package/backend/src/flowent/static/assets/geist-latin-wght-normal-Dm3htQBi.woff2 +0 -0
  28. package/backend/src/flowent/static/assets/index-C76K95ty.js +81 -0
  29. package/backend/src/flowent/static/assets/index-iUMNKvlU.css +2 -0
  30. package/backend/src/flowent/static/flowent.png +0 -0
  31. package/backend/src/flowent/static/index.html +5 -25
  32. package/backend/src/flowent/storage.py +302 -0
  33. package/backend/src/flowent/tools.py +376 -0
  34. package/backend/tests/__pycache__/test_agent_tools.cpython-313-pytest-9.0.3.pyc +0 -0
  35. package/backend/tests/__pycache__/test_health.cpython-313-pytest-9.0.3.pyc +0 -0
  36. package/backend/tests/__pycache__/test_llm_providers.cpython-313-pytest-9.0.3.pyc +0 -0
  37. package/backend/tests/__pycache__/test_logging.cpython-313-pytest-9.0.3.pyc +0 -0
  38. package/backend/tests/__pycache__/test_persistence.cpython-313-pytest-9.0.3.pyc +0 -0
  39. package/backend/tests/__pycache__/test_workspace_chat.cpython-313-pytest-9.0.3.pyc +0 -0
  40. package/backend/tests/test_agent_tools.py +477 -0
  41. package/backend/tests/test_health.py +12 -0
  42. package/backend/tests/test_llm_providers.py +113 -0
  43. package/backend/tests/test_logging.py +182 -0
  44. package/backend/tests/test_persistence.py +125 -0
  45. package/backend/tests/test_workspace_chat.py +578 -0
  46. package/backend/uv.lock +803 -99
  47. package/dist/frontend/assets/geist-cyrillic-wght-normal-CHSlOQsW.woff2 +0 -0
  48. package/dist/frontend/assets/geist-latin-ext-wght-normal-DMtmJ5ZE.woff2 +0 -0
  49. package/dist/frontend/assets/geist-latin-wght-normal-Dm3htQBi.woff2 +0 -0
  50. package/dist/frontend/assets/index-C76K95ty.js +81 -0
  51. package/dist/frontend/assets/index-iUMNKvlU.css +2 -0
  52. package/dist/frontend/flowent.png +0 -0
  53. package/dist/frontend/index.html +5 -25
  54. package/package.json +1 -2
  55. package/backend/src/flowent/__pycache__/_version.cpython-313.pyc +0 -0
  56. package/backend/src/flowent/__pycache__/access.cpython-313.pyc +0 -0
  57. package/backend/src/flowent/__pycache__/assistant_commands.cpython-313.pyc +0 -0
  58. package/backend/src/flowent/__pycache__/cli.cpython-313.pyc +0 -0
  59. package/backend/src/flowent/__pycache__/config.cpython-313.pyc +0 -0
  60. package/backend/src/flowent/__pycache__/events.cpython-313.pyc +0 -0
  61. package/backend/src/flowent/__pycache__/graph_runtime.cpython-313.pyc +0 -0
  62. package/backend/src/flowent/__pycache__/graph_service.cpython-313.pyc +0 -0
  63. package/backend/src/flowent/__pycache__/image_assets.cpython-313.pyc +0 -0
  64. package/backend/src/flowent/__pycache__/model_metadata.cpython-313.pyc +0 -0
  65. package/backend/src/flowent/__pycache__/network.cpython-313.pyc +0 -0
  66. package/backend/src/flowent/__pycache__/observability_service.cpython-313.pyc +0 -0
  67. package/backend/src/flowent/__pycache__/registry.cpython-313.pyc +0 -0
  68. package/backend/src/flowent/__pycache__/role_management.cpython-313.pyc +0 -0
  69. package/backend/src/flowent/__pycache__/runtime.cpython-313.pyc +0 -0
  70. package/backend/src/flowent/__pycache__/security.cpython-313.pyc +0 -0
  71. package/backend/src/flowent/__pycache__/settings.cpython-313.pyc +0 -0
  72. package/backend/src/flowent/__pycache__/settings_management.cpython-313.pyc +0 -0
  73. package/backend/src/flowent/__pycache__/state_db.cpython-313.pyc +0 -0
  74. package/backend/src/flowent/__pycache__/workspace_store.cpython-313.pyc +0 -0
  75. package/backend/src/flowent/access.py +0 -247
  76. package/backend/src/flowent/assistant_commands.py +0 -115
  77. package/backend/src/flowent/channels/__init__.py +0 -3
  78. package/backend/src/flowent/channels/__pycache__/__init__.cpython-313.pyc +0 -0
  79. package/backend/src/flowent/channels/__pycache__/telegram.cpython-313.pyc +0 -0
  80. package/backend/src/flowent/channels/telegram.py +0 -615
  81. package/backend/src/flowent/config.py +0 -14
  82. package/backend/src/flowent/dev.py +0 -3
  83. package/backend/src/flowent/events.py +0 -157
  84. package/backend/src/flowent/graph_runtime.py +0 -60
  85. package/backend/src/flowent/graph_service.py +0 -2401
  86. package/backend/src/flowent/image_assets.py +0 -356
  87. package/backend/src/flowent/model_metadata.py +0 -102
  88. package/backend/src/flowent/models/__init__.py +0 -125
  89. package/backend/src/flowent/models/__pycache__/__init__.cpython-313.pyc +0 -0
  90. package/backend/src/flowent/models/__pycache__/agent.cpython-313.pyc +0 -0
  91. package/backend/src/flowent/models/__pycache__/base.cpython-313.pyc +0 -0
  92. package/backend/src/flowent/models/__pycache__/blueprint.cpython-313.pyc +0 -0
  93. package/backend/src/flowent/models/__pycache__/content.cpython-313.pyc +0 -0
  94. package/backend/src/flowent/models/__pycache__/delta.cpython-313.pyc +0 -0
  95. package/backend/src/flowent/models/__pycache__/event.cpython-313.pyc +0 -0
  96. package/backend/src/flowent/models/__pycache__/graph.cpython-313.pyc +0 -0
  97. package/backend/src/flowent/models/__pycache__/history.cpython-313.pyc +0 -0
  98. package/backend/src/flowent/models/__pycache__/llm.cpython-313.pyc +0 -0
  99. package/backend/src/flowent/models/__pycache__/message.cpython-313.pyc +0 -0
  100. package/backend/src/flowent/models/__pycache__/tab.cpython-313.pyc +0 -0
  101. package/backend/src/flowent/models/__pycache__/todo.cpython-313.pyc +0 -0
  102. package/backend/src/flowent/models/agent.py +0 -34
  103. package/backend/src/flowent/models/base.py +0 -24
  104. package/backend/src/flowent/models/blueprint.py +0 -176
  105. package/backend/src/flowent/models/content.py +0 -164
  106. package/backend/src/flowent/models/delta.py +0 -44
  107. package/backend/src/flowent/models/event.py +0 -51
  108. package/backend/src/flowent/models/graph.py +0 -472
  109. package/backend/src/flowent/models/history.py +0 -272
  110. package/backend/src/flowent/models/llm.py +0 -62
  111. package/backend/src/flowent/models/message.py +0 -33
  112. package/backend/src/flowent/models/tab.py +0 -85
  113. package/backend/src/flowent/models/todo.py +0 -10
  114. package/backend/src/flowent/network.py +0 -146
  115. package/backend/src/flowent/observability_service.py +0 -218
  116. package/backend/src/flowent/prompts/__init__.py +0 -67
  117. package/backend/src/flowent/prompts/__pycache__/__init__.cpython-313.pyc +0 -0
  118. package/backend/src/flowent/prompts/__pycache__/common.cpython-313.pyc +0 -0
  119. package/backend/src/flowent/prompts/__pycache__/steward.cpython-313.pyc +0 -0
  120. package/backend/src/flowent/prompts/common.py +0 -250
  121. package/backend/src/flowent/prompts/steward.py +0 -64
  122. package/backend/src/flowent/providers/__init__.py +0 -23
  123. package/backend/src/flowent/providers/__pycache__/__init__.cpython-313.pyc +0 -0
  124. package/backend/src/flowent/providers/__pycache__/anthropic.cpython-313.pyc +0 -0
  125. package/backend/src/flowent/providers/__pycache__/base_url.cpython-313.pyc +0 -0
  126. package/backend/src/flowent/providers/__pycache__/configuration.cpython-313.pyc +0 -0
  127. package/backend/src/flowent/providers/__pycache__/content.cpython-313.pyc +0 -0
  128. package/backend/src/flowent/providers/__pycache__/errors.cpython-313.pyc +0 -0
  129. package/backend/src/flowent/providers/__pycache__/gateway.cpython-313.pyc +0 -0
  130. package/backend/src/flowent/providers/__pycache__/headers.cpython-313.pyc +0 -0
  131. package/backend/src/flowent/providers/__pycache__/management.cpython-313.pyc +0 -0
  132. package/backend/src/flowent/providers/__pycache__/openai.cpython-313.pyc +0 -0
  133. package/backend/src/flowent/providers/__pycache__/openai_responses.cpython-313.pyc +0 -0
  134. package/backend/src/flowent/providers/__pycache__/registry.cpython-313.pyc +0 -0
  135. package/backend/src/flowent/providers/__pycache__/sse.cpython-313.pyc +0 -0
  136. package/backend/src/flowent/providers/__pycache__/thinking.cpython-313.pyc +0 -0
  137. package/backend/src/flowent/providers/anthropic.py +0 -468
  138. package/backend/src/flowent/providers/base_url.py +0 -60
  139. package/backend/src/flowent/providers/configuration.py +0 -189
  140. package/backend/src/flowent/providers/content.py +0 -122
  141. package/backend/src/flowent/providers/errors.py +0 -223
  142. package/backend/src/flowent/providers/gateway.py +0 -169
  143. package/backend/src/flowent/providers/gemini.py +0 -447
  144. package/backend/src/flowent/providers/headers.py +0 -20
  145. package/backend/src/flowent/providers/management.py +0 -96
  146. package/backend/src/flowent/providers/ollama.py +0 -293
  147. package/backend/src/flowent/providers/openai.py +0 -422
  148. package/backend/src/flowent/providers/openai_responses.py +0 -655
  149. package/backend/src/flowent/providers/registry.py +0 -144
  150. package/backend/src/flowent/providers/sse.py +0 -31
  151. package/backend/src/flowent/providers/thinking.py +0 -79
  152. package/backend/src/flowent/registry.py +0 -73
  153. package/backend/src/flowent/role_management.py +0 -270
  154. package/backend/src/flowent/routes/__init__.py +0 -26
  155. package/backend/src/flowent/routes/__pycache__/__init__.cpython-313.pyc +0 -0
  156. package/backend/src/flowent/routes/__pycache__/access.cpython-313.pyc +0 -0
  157. package/backend/src/flowent/routes/__pycache__/assistant.cpython-313.pyc +0 -0
  158. package/backend/src/flowent/routes/__pycache__/image_assets.cpython-313.pyc +0 -0
  159. package/backend/src/flowent/routes/__pycache__/meta.cpython-313.pyc +0 -0
  160. package/backend/src/flowent/routes/__pycache__/nodes.cpython-313.pyc +0 -0
  161. package/backend/src/flowent/routes/__pycache__/prompts.cpython-313.pyc +0 -0
  162. package/backend/src/flowent/routes/__pycache__/providers_route.cpython-313.pyc +0 -0
  163. package/backend/src/flowent/routes/__pycache__/roles.cpython-313.pyc +0 -0
  164. package/backend/src/flowent/routes/__pycache__/settings.cpython-313.pyc +0 -0
  165. package/backend/src/flowent/routes/__pycache__/tabs.cpython-313.pyc +0 -0
  166. package/backend/src/flowent/routes/__pycache__/ws.cpython-313.pyc +0 -0
  167. package/backend/src/flowent/routes/access.py +0 -48
  168. package/backend/src/flowent/routes/assistant.py +0 -158
  169. package/backend/src/flowent/routes/image_assets.py +0 -33
  170. package/backend/src/flowent/routes/meta.py +0 -28
  171. package/backend/src/flowent/routes/nodes.py +0 -423
  172. package/backend/src/flowent/routes/prompts.py +0 -46
  173. package/backend/src/flowent/routes/providers_route.py +0 -365
  174. package/backend/src/flowent/routes/roles.py +0 -207
  175. package/backend/src/flowent/routes/settings.py +0 -379
  176. package/backend/src/flowent/routes/tabs.py +0 -298
  177. package/backend/src/flowent/routes/ws.py +0 -33
  178. package/backend/src/flowent/runtime.py +0 -160
  179. package/backend/src/flowent/security.py +0 -37
  180. package/backend/src/flowent/settings.py +0 -2112
  181. package/backend/src/flowent/settings_management.py +0 -394
  182. package/backend/src/flowent/state_db.py +0 -108
  183. package/backend/src/flowent/static/assets/AssistantPage-BW7XAd9I.js +0 -1
  184. package/backend/src/flowent/static/assets/ChannelsPage-tCJHgt6m.js +0 -1
  185. package/backend/src/flowent/static/assets/PageScaffold-f6g2l7XN.js +0 -1
  186. package/backend/src/flowent/static/assets/PromptsPage-C3Sxn2D7.js +0 -1
  187. package/backend/src/flowent/static/assets/ProvidersPage-BfmdXmNt.js +0 -3
  188. package/backend/src/flowent/static/assets/RolesPage-DET8wO4r.js +0 -1
  189. package/backend/src/flowent/static/assets/SettingsPage-D-g3deMm.js +0 -3
  190. package/backend/src/flowent/static/assets/ToolsPage-CDmtE2g4.js +0 -1
  191. package/backend/src/flowent/static/assets/WorkspacePage-AZsJ0sD0.js +0 -3
  192. package/backend/src/flowent/static/assets/WorkspacePanels-CteCjolX.js +0 -1
  193. package/backend/src/flowent/static/assets/alert-dialog-Duorp_S-.js +0 -1
  194. package/backend/src/flowent/static/assets/dialog-C3ixjGjN.js +0 -1
  195. package/backend/src/flowent/static/assets/elk-worker.min-C9JGDOE-.js +0 -6312
  196. package/backend/src/flowent/static/assets/graph-vendor-CHpVij2M.css +0 -1
  197. package/backend/src/flowent/static/assets/graph-vendor-DRq_-6fV.js +0 -7
  198. package/backend/src/flowent/static/assets/index--o_0fv0N.css +0 -1
  199. package/backend/src/flowent/static/assets/index-C9HuekJm.js +0 -10
  200. package/backend/src/flowent/static/assets/layout.worker-jMHqAFbP.js +0 -24
  201. package/backend/src/flowent/static/assets/markdown-vendor-C9RtvaJh.js +0 -29
  202. package/backend/src/flowent/static/assets/modelParams-DmnF2hwR.js +0 -1
  203. package/backend/src/flowent/static/assets/providerTypes-DT3Ahwl_.js +0 -1
  204. package/backend/src/flowent/static/assets/react-vendor-mEs_JJxa.js +0 -9
  205. package/backend/src/flowent/static/assets/roles-CuRT_chR.js +0 -1
  206. package/backend/src/flowent/static/assets/rolldown-runtime-BYbx6iT9.js +0 -1
  207. package/backend/src/flowent/static/assets/select-DCfeNu-F.js +0 -1
  208. package/backend/src/flowent/static/assets/surface-pWwG5ogx.js +0 -1
  209. package/backend/src/flowent/static/assets/ui-vendor-C5pJa8N7.js +0 -51
  210. package/backend/src/flowent/static/assets/useAppRoute-FgSHBKhV.js +0 -1
  211. package/backend/src/flowent/static/favicon.svg +0 -4
  212. package/backend/src/flowent/tools/__init__.py +0 -176
  213. package/backend/src/flowent/tools/__pycache__/__init__.cpython-313.pyc +0 -0
  214. package/backend/src/flowent/tools/__pycache__/connect.cpython-313.pyc +0 -0
  215. package/backend/src/flowent/tools/__pycache__/contacts.cpython-313.pyc +0 -0
  216. package/backend/src/flowent/tools/__pycache__/create_agent.cpython-313.pyc +0 -0
  217. package/backend/src/flowent/tools/__pycache__/create_tab.cpython-313.pyc +0 -0
  218. package/backend/src/flowent/tools/__pycache__/delete_tab.cpython-313.pyc +0 -0
  219. package/backend/src/flowent/tools/__pycache__/edit.cpython-313.pyc +0 -0
  220. package/backend/src/flowent/tools/__pycache__/exec.cpython-313.pyc +0 -0
  221. package/backend/src/flowent/tools/__pycache__/fetch.cpython-313.pyc +0 -0
  222. package/backend/src/flowent/tools/__pycache__/idle.cpython-313.pyc +0 -0
  223. package/backend/src/flowent/tools/__pycache__/list_roles.cpython-313.pyc +0 -0
  224. package/backend/src/flowent/tools/__pycache__/list_tabs.cpython-313.pyc +0 -0
  225. package/backend/src/flowent/tools/__pycache__/list_tools.cpython-313.pyc +0 -0
  226. package/backend/src/flowent/tools/__pycache__/manage_prompts.cpython-313.pyc +0 -0
  227. package/backend/src/flowent/tools/__pycache__/manage_providers.cpython-313.pyc +0 -0
  228. package/backend/src/flowent/tools/__pycache__/manage_roles.cpython-313.pyc +0 -0
  229. package/backend/src/flowent/tools/__pycache__/manage_settings.cpython-313.pyc +0 -0
  230. package/backend/src/flowent/tools/__pycache__/read.cpython-313.pyc +0 -0
  231. package/backend/src/flowent/tools/__pycache__/send.cpython-313.pyc +0 -0
  232. package/backend/src/flowent/tools/__pycache__/set_permissions.cpython-313.pyc +0 -0
  233. package/backend/src/flowent/tools/__pycache__/sleep.cpython-313.pyc +0 -0
  234. package/backend/src/flowent/tools/__pycache__/todo.cpython-313.pyc +0 -0
  235. package/backend/src/flowent/tools/connect.py +0 -100
  236. package/backend/src/flowent/tools/contacts.py +0 -22
  237. package/backend/src/flowent/tools/create_agent.py +0 -191
  238. package/backend/src/flowent/tools/create_tab.py +0 -61
  239. package/backend/src/flowent/tools/delete_tab.py +0 -39
  240. package/backend/src/flowent/tools/edit.py +0 -142
  241. package/backend/src/flowent/tools/exec.py +0 -118
  242. package/backend/src/flowent/tools/fetch.py +0 -85
  243. package/backend/src/flowent/tools/idle.py +0 -27
  244. package/backend/src/flowent/tools/list_roles.py +0 -68
  245. package/backend/src/flowent/tools/list_tabs.py +0 -100
  246. package/backend/src/flowent/tools/list_tools.py +0 -28
  247. package/backend/src/flowent/tools/manage_prompts.py +0 -102
  248. package/backend/src/flowent/tools/manage_providers.py +0 -220
  249. package/backend/src/flowent/tools/manage_roles.py +0 -275
  250. package/backend/src/flowent/tools/manage_settings.py +0 -326
  251. package/backend/src/flowent/tools/read.py +0 -152
  252. package/backend/src/flowent/tools/send.py +0 -68
  253. package/backend/src/flowent/tools/set_permissions.py +0 -99
  254. package/backend/src/flowent/tools/sleep.py +0 -41
  255. package/backend/src/flowent/tools/todo.py +0 -51
  256. package/backend/src/flowent/workspace_store.py +0 -479
  257. package/backend/tests/__init__.py +0 -0
  258. package/backend/tests/__pycache__/__init__.cpython-313.pyc +0 -0
  259. package/backend/tests/__pycache__/conftest.cpython-313-pytest-9.0.3.pyc +0 -0
  260. package/backend/tests/conftest.py +0 -6
  261. package/backend/tests/integration/api/__pycache__/conftest.cpython-313-pytest-9.0.3.pyc +0 -0
  262. package/backend/tests/integration/api/__pycache__/test_access_api.cpython-313-pytest-9.0.3.pyc +0 -0
  263. package/backend/tests/integration/api/__pycache__/test_assistant_api.cpython-313-pytest-9.0.3.pyc +0 -0
  264. package/backend/tests/integration/api/__pycache__/test_frontend_mounting.cpython-313-pytest-9.0.3.pyc +0 -0
  265. package/backend/tests/integration/api/__pycache__/test_meta_api.cpython-313-pytest-9.0.3.pyc +0 -0
  266. package/backend/tests/integration/api/__pycache__/test_nodes_api.cpython-313-pytest-9.0.3.pyc +0 -0
  267. package/backend/tests/integration/api/__pycache__/test_prompts_api.cpython-313-pytest-9.0.3.pyc +0 -0
  268. package/backend/tests/integration/api/__pycache__/test_roles_api.cpython-313-pytest-9.0.3.pyc +0 -0
  269. package/backend/tests/integration/api/__pycache__/test_tabs_api.cpython-313-pytest-9.0.3.pyc +0 -0
  270. package/backend/tests/integration/api/conftest.py +0 -29
  271. package/backend/tests/integration/api/test_access_api.py +0 -182
  272. package/backend/tests/integration/api/test_assistant_api.py +0 -422
  273. package/backend/tests/integration/api/test_frontend_mounting.py +0 -61
  274. package/backend/tests/integration/api/test_meta_api.py +0 -32
  275. package/backend/tests/integration/api/test_nodes_api.py +0 -787
  276. package/backend/tests/integration/api/test_prompts_api.py +0 -47
  277. package/backend/tests/integration/api/test_roles_api.py +0 -228
  278. package/backend/tests/integration/api/test_tabs_api.py +0 -688
  279. package/backend/tests/unit/__pycache__/test_access.cpython-313-pytest-9.0.3.pyc +0 -0
  280. package/backend/tests/unit/__pycache__/test_cli.cpython-313-pytest-9.0.3.pyc +0 -0
  281. package/backend/tests/unit/__pycache__/test_graph_runtime.cpython-313-pytest-9.0.3.pyc +0 -0
  282. package/backend/tests/unit/__pycache__/test_network.cpython-313-pytest-9.0.3.pyc +0 -0
  283. package/backend/tests/unit/__pycache__/test_state_sqlite_storage.cpython-313-pytest-9.0.3.pyc +0 -0
  284. package/backend/tests/unit/__pycache__/test_workspace_store.cpython-313-pytest-9.0.3.pyc +0 -0
  285. package/backend/tests/unit/agent/__pycache__/test_agent_public_api.cpython-313-pytest-9.0.3.pyc +0 -0
  286. package/backend/tests/unit/agent/__pycache__/test_agent_runtime.cpython-313-pytest-9.0.3.pyc +0 -0
  287. package/backend/tests/unit/agent/test_agent_public_api.py +0 -822
  288. package/backend/tests/unit/agent/test_agent_runtime.py +0 -3088
  289. package/backend/tests/unit/channels/__pycache__/test_telegram_channel.cpython-313-pytest-9.0.3.pyc +0 -0
  290. package/backend/tests/unit/channels/test_telegram_channel.py +0 -552
  291. package/backend/tests/unit/logging/__pycache__/test_logging.cpython-313-pytest-9.0.3.pyc +0 -0
  292. package/backend/tests/unit/logging/test_logging.py +0 -132
  293. package/backend/tests/unit/prompts/__pycache__/test_prompts.cpython-313-pytest-9.0.3.pyc +0 -0
  294. package/backend/tests/unit/prompts/test_prompts.py +0 -570
  295. package/backend/tests/unit/providers/__pycache__/test_anthropic_provider.cpython-313-pytest-9.0.3.pyc +0 -0
  296. package/backend/tests/unit/providers/__pycache__/test_errors.cpython-313-pytest-9.0.3.pyc +0 -0
  297. package/backend/tests/unit/providers/__pycache__/test_extract_delta_parts.cpython-313-pytest-9.0.3.pyc +0 -0
  298. package/backend/tests/unit/providers/__pycache__/test_openai_provider.cpython-313-pytest-9.0.3.pyc +0 -0
  299. package/backend/tests/unit/providers/__pycache__/test_openai_responses.cpython-313-pytest-9.0.3.pyc +0 -0
  300. package/backend/tests/unit/providers/__pycache__/test_provider_gateway.cpython-313-pytest-9.0.3.pyc +0 -0
  301. package/backend/tests/unit/providers/__pycache__/test_think_tag_parser.cpython-313-pytest-9.0.3.pyc +0 -0
  302. package/backend/tests/unit/providers/test_anthropic_provider.py +0 -185
  303. package/backend/tests/unit/providers/test_errors.py +0 -68
  304. package/backend/tests/unit/providers/test_extract_delta_parts.py +0 -22
  305. package/backend/tests/unit/providers/test_openai_provider.py +0 -139
  306. package/backend/tests/unit/providers/test_openai_responses.py +0 -402
  307. package/backend/tests/unit/providers/test_provider_gateway.py +0 -359
  308. package/backend/tests/unit/providers/test_think_tag_parser.py +0 -36
  309. package/backend/tests/unit/routes/__pycache__/test_prompts_routes.cpython-313-pytest-9.0.3.pyc +0 -0
  310. package/backend/tests/unit/routes/__pycache__/test_providers_route.cpython-313-pytest-9.0.3.pyc +0 -0
  311. package/backend/tests/unit/routes/__pycache__/test_roles_routes.cpython-313-pytest-9.0.3.pyc +0 -0
  312. package/backend/tests/unit/routes/__pycache__/test_settings_routes.cpython-313-pytest-9.0.3.pyc +0 -0
  313. package/backend/tests/unit/routes/test_prompts_routes.py +0 -82
  314. package/backend/tests/unit/routes/test_providers_route.py +0 -370
  315. package/backend/tests/unit/routes/test_roles_routes.py +0 -539
  316. package/backend/tests/unit/routes/test_settings_routes.py +0 -1123
  317. package/backend/tests/unit/runtime/__pycache__/test_bootstrap_runtime.cpython-313-pytest-9.0.3.pyc +0 -0
  318. package/backend/tests/unit/runtime/test_bootstrap_runtime.py +0 -1002
  319. package/backend/tests/unit/sandbox/__pycache__/test_sandbox_tools.cpython-313-pytest-9.0.3.pyc +0 -0
  320. package/backend/tests/unit/sandbox/test_sandbox_tools.py +0 -78
  321. package/backend/tests/unit/security/__pycache__/test_security.cpython-313-pytest-9.0.3.pyc +0 -0
  322. package/backend/tests/unit/security/test_security.py +0 -124
  323. package/backend/tests/unit/settings/__pycache__/test_settings_roles.cpython-313-pytest-9.0.3.pyc +0 -0
  324. package/backend/tests/unit/settings/test_settings_roles.py +0 -703
  325. package/backend/tests/unit/test_access.py +0 -45
  326. package/backend/tests/unit/test_cli.py +0 -102
  327. package/backend/tests/unit/test_graph_runtime.py +0 -72
  328. package/backend/tests/unit/test_network.py +0 -51
  329. package/backend/tests/unit/test_state_sqlite_storage.py +0 -87
  330. package/backend/tests/unit/test_workspace_store.py +0 -228
  331. package/backend/tests/unit/tools/__pycache__/test_connect_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  332. package/backend/tests/unit/tools/__pycache__/test_create_agent_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  333. package/backend/tests/unit/tools/__pycache__/test_delete_tab_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  334. package/backend/tests/unit/tools/__pycache__/test_edit_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  335. package/backend/tests/unit/tools/__pycache__/test_exec_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  336. package/backend/tests/unit/tools/__pycache__/test_fetch_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  337. package/backend/tests/unit/tools/__pycache__/test_manage_prompts_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  338. package/backend/tests/unit/tools/__pycache__/test_manage_providers_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  339. package/backend/tests/unit/tools/__pycache__/test_manage_roles_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  340. package/backend/tests/unit/tools/__pycache__/test_manage_settings_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  341. package/backend/tests/unit/tools/__pycache__/test_read_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  342. package/backend/tests/unit/tools/__pycache__/test_set_permissions_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  343. package/backend/tests/unit/tools/__pycache__/test_todo_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  344. package/backend/tests/unit/tools/__pycache__/test_tool_registry.cpython-313-pytest-9.0.3.pyc +0 -0
  345. package/backend/tests/unit/tools/test_connect_tool.py +0 -228
  346. package/backend/tests/unit/tools/test_create_agent_tool.py +0 -404
  347. package/backend/tests/unit/tools/test_delete_tab_tool.py +0 -116
  348. package/backend/tests/unit/tools/test_edit_tool.py +0 -115
  349. package/backend/tests/unit/tools/test_exec_tool.py +0 -81
  350. package/backend/tests/unit/tools/test_fetch_tool.py +0 -65
  351. package/backend/tests/unit/tools/test_manage_prompts_tool.py +0 -92
  352. package/backend/tests/unit/tools/test_manage_providers_tool.py +0 -460
  353. package/backend/tests/unit/tools/test_manage_roles_tool.py +0 -411
  354. package/backend/tests/unit/tools/test_manage_settings_tool.py +0 -611
  355. package/backend/tests/unit/tools/test_read_tool.py +0 -33
  356. package/backend/tests/unit/tools/test_set_permissions_tool.py +0 -595
  357. package/backend/tests/unit/tools/test_todo_tool.py +0 -37
  358. package/backend/tests/unit/tools/test_tool_registry.py +0 -199
  359. package/dist/frontend/assets/AssistantPage-BW7XAd9I.js +0 -1
  360. package/dist/frontend/assets/ChannelsPage-tCJHgt6m.js +0 -1
  361. package/dist/frontend/assets/PageScaffold-f6g2l7XN.js +0 -1
  362. package/dist/frontend/assets/PromptsPage-C3Sxn2D7.js +0 -1
  363. package/dist/frontend/assets/ProvidersPage-BfmdXmNt.js +0 -3
  364. package/dist/frontend/assets/RolesPage-DET8wO4r.js +0 -1
  365. package/dist/frontend/assets/SettingsPage-D-g3deMm.js +0 -3
  366. package/dist/frontend/assets/ToolsPage-CDmtE2g4.js +0 -1
  367. package/dist/frontend/assets/WorkspacePage-AZsJ0sD0.js +0 -3
  368. package/dist/frontend/assets/WorkspacePanels-CteCjolX.js +0 -1
  369. package/dist/frontend/assets/alert-dialog-Duorp_S-.js +0 -1
  370. package/dist/frontend/assets/dialog-C3ixjGjN.js +0 -1
  371. package/dist/frontend/assets/elk-worker.min-C9JGDOE-.js +0 -6312
  372. package/dist/frontend/assets/graph-vendor-CHpVij2M.css +0 -1
  373. package/dist/frontend/assets/graph-vendor-DRq_-6fV.js +0 -7
  374. package/dist/frontend/assets/index--o_0fv0N.css +0 -1
  375. package/dist/frontend/assets/index-C9HuekJm.js +0 -10
  376. package/dist/frontend/assets/layout.worker-jMHqAFbP.js +0 -24
  377. package/dist/frontend/assets/markdown-vendor-C9RtvaJh.js +0 -29
  378. package/dist/frontend/assets/modelParams-DmnF2hwR.js +0 -1
  379. package/dist/frontend/assets/providerTypes-DT3Ahwl_.js +0 -1
  380. package/dist/frontend/assets/react-vendor-mEs_JJxa.js +0 -9
  381. package/dist/frontend/assets/roles-CuRT_chR.js +0 -1
  382. package/dist/frontend/assets/rolldown-runtime-BYbx6iT9.js +0 -1
  383. package/dist/frontend/assets/select-DCfeNu-F.js +0 -1
  384. package/dist/frontend/assets/surface-pWwG5ogx.js +0 -1
  385. package/dist/frontend/assets/ui-vendor-C5pJa8N7.js +0 -51
  386. package/dist/frontend/assets/useAppRoute-FgSHBKhV.js +0 -1
  387. package/dist/frontend/favicon.svg +0 -4
@@ -0,0 +1,477 @@
1
+ import json
2
+ from pathlib import Path
3
+
4
+ from fastapi.testclient import TestClient
5
+
6
+ from flowent.agent import FLOWENT_AGENT_SYSTEM_PROMPT
7
+ from flowent.main import create_app
8
+ from flowent.sandbox import SandboxRunner
9
+ from flowent.tools import ToolContext, run_tool
10
+
11
+
12
+ def stream_events(content: str) -> list[dict[str, object]]:
13
+ events: list[dict[str, object]] = []
14
+ for raw_event in content.strip().split("\n\n"):
15
+ event_type = ""
16
+ data = ""
17
+ for line in raw_event.splitlines():
18
+ if line.startswith("event: "):
19
+ event_type = line.removeprefix("event: ")
20
+ if line.startswith("data: "):
21
+ data = line.removeprefix("data: ")
22
+ events.append({"event": event_type, "data": json.loads(data)})
23
+ return events
24
+
25
+
26
+ def configure_provider(client: TestClient) -> None:
27
+ client.post(
28
+ "/api/providers",
29
+ json={
30
+ "api_key": "sk-local",
31
+ "base_url": "",
32
+ "id": "provider-openai",
33
+ "models": ["gpt-5.1"],
34
+ "name": "OpenAI",
35
+ "type": "openai",
36
+ },
37
+ )
38
+ client.put(
39
+ "/api/settings",
40
+ json={
41
+ "selected_model": "gpt-5.1",
42
+ "selected_provider_id": "provider-openai",
43
+ },
44
+ )
45
+
46
+
47
+ def tool_call_chunk(
48
+ name: str, arguments: dict[str, object], call_id: str = "call-1"
49
+ ) -> dict[str, object]:
50
+ return {
51
+ "choices": [
52
+ {
53
+ "delta": {
54
+ "tool_calls": [
55
+ {
56
+ "index": 0,
57
+ "id": call_id,
58
+ "type": "function",
59
+ "function": {
60
+ "name": name,
61
+ "arguments": json.dumps(arguments),
62
+ },
63
+ }
64
+ ]
65
+ }
66
+ }
67
+ ]
68
+ }
69
+
70
+
71
+ def text_chunk(content: str) -> dict[str, object]:
72
+ return {"choices": [{"delta": {"content": content}}]}
73
+
74
+
75
+ def test_workspace_response_streams_tool_process_and_final_text(
76
+ tmp_path, monkeypatch
77
+ ) -> None:
78
+ monkeypatch.setenv("FLOWENT_DATA_DIR", str(tmp_path / "data"))
79
+ workdir = tmp_path / "workdir"
80
+ workdir.mkdir()
81
+ (workdir / "notes.txt").write_text("Launch notes")
82
+ monkeypatch.chdir(workdir)
83
+ captured_requests: list[dict[str, object]] = []
84
+
85
+ async def fake_completion(**request: object) -> object:
86
+ captured_requests.append(request)
87
+
88
+ async def chunks() -> object:
89
+ if len(captured_requests) == 1:
90
+ yield tool_call_chunk("read_file", {"path": "notes.txt"})
91
+ else:
92
+ yield text_chunk("Read the notes.")
93
+
94
+ return chunks()
95
+
96
+ client = TestClient(
97
+ create_app(serve_frontend=False, chat_completion=fake_completion)
98
+ )
99
+ configure_provider(client)
100
+
101
+ response = client.post(
102
+ "/api/workspace/respond",
103
+ json={"content": "Use the notes."},
104
+ )
105
+
106
+ assert response.status_code == 200
107
+ events = stream_events(response.text)
108
+ assert [event["event"] for event in events] == [
109
+ "start",
110
+ "output_start",
111
+ "tool_start",
112
+ "tool_done",
113
+ "output_start",
114
+ "delta",
115
+ "done",
116
+ ]
117
+ assert events[1]["data"] == {"index": 1}
118
+ assert events[2]["data"]["tool"]["status"] == "running"
119
+ assert events[3]["data"]["status"] == "success"
120
+ assert events[4]["data"] == {"index": 2}
121
+ assert events[5]["data"] == {"content": "Read the notes."}
122
+ assert events[6]["data"]["message"]["content"] == "Read the notes."
123
+ assert len(captured_requests) == 2
124
+ assert captured_requests[0]["messages"][0] == {
125
+ "role": "system",
126
+ "content": FLOWENT_AGENT_SYSTEM_PROMPT,
127
+ }
128
+ second_messages = captured_requests[1]["messages"]
129
+ assert second_messages[0] == {
130
+ "role": "system",
131
+ "content": FLOWENT_AGENT_SYSTEM_PROMPT,
132
+ }
133
+ assert second_messages[-2]["tool_calls"][0]["function"]["name"] == "read_file"
134
+ assert second_messages[-1] == {
135
+ "role": "tool",
136
+ "tool_call_id": "call-1",
137
+ "content": "Launch notes",
138
+ }
139
+
140
+
141
+ def test_tools_can_read_paths_outside_workdir(tmp_path) -> None:
142
+ outside = tmp_path / "outside.txt"
143
+ outside.write_text("outside content")
144
+
145
+ result = run_tool(
146
+ "read_file", {"path": str(outside)}, ToolContext(cwd=tmp_path / "work")
147
+ )
148
+
149
+ assert result.ok
150
+ assert result.content == "outside content"
151
+
152
+
153
+ def test_list_dir_can_list_paths_outside_workdir(tmp_path) -> None:
154
+ outside = tmp_path / "outside"
155
+ outside.mkdir()
156
+ (outside / "file.txt").write_text("content")
157
+
158
+ result = run_tool(
159
+ "list_dir", {"path": str(outside)}, ToolContext(cwd=tmp_path / "work")
160
+ )
161
+
162
+ assert result.ok
163
+ assert "file.txt" in result.content
164
+
165
+
166
+ def test_grep_files_can_search_paths_outside_workdir(tmp_path) -> None:
167
+ outside = tmp_path / "outside"
168
+ outside.mkdir()
169
+ (outside / "file.txt").write_text("alpha beta")
170
+
171
+ result = run_tool(
172
+ "grep_files",
173
+ {"pattern": "alpha", "path": str(outside)},
174
+ ToolContext(cwd=tmp_path / "work"),
175
+ )
176
+
177
+ assert result.ok
178
+ assert "file.txt" in result.content
179
+
180
+
181
+ def test_shell_command_can_write_workdir_and_tmp(tmp_path) -> None:
182
+ result = run_tool(
183
+ "shell_command",
184
+ {"command": "echo ok > work.txt && echo tmp > /tmp/flowent-tool-test.txt"},
185
+ ToolContext(cwd=tmp_path),
186
+ )
187
+
188
+ assert result.ok
189
+ assert (tmp_path / "work.txt").read_text().strip() == "ok"
190
+
191
+
192
+ def test_shell_command_cannot_write_outside_workdir_and_tmp(tmp_path) -> None:
193
+ outside = Path("/project/flowent/backend/tests/flowent-outside-denied.txt")
194
+ if outside.exists():
195
+ outside.unlink()
196
+
197
+ result = run_tool(
198
+ "shell_command",
199
+ {"command": f"echo denied > {outside}"},
200
+ ToolContext(cwd=tmp_path),
201
+ )
202
+
203
+ assert not result.ok
204
+ assert not outside.exists()
205
+
206
+
207
+ def test_shell_command_has_network_by_default(tmp_path) -> None:
208
+ result = run_tool(
209
+ "shell_command",
210
+ {
211
+ "command": "python - <<'PY'\nimport socket\ns=socket.socket()\nprint('network-ready')\nPY"
212
+ },
213
+ ToolContext(cwd=tmp_path),
214
+ )
215
+
216
+ assert result.ok
217
+ assert "network-ready" in result.content
218
+
219
+
220
+ def test_shell_command_denies_ptrace_when_seccomp_is_available(tmp_path) -> None:
221
+ command = SandboxRunner(cwd=tmp_path).build_command(["/bin/true"])
222
+ if not command.seccomp_available:
223
+ assert command.args[0].endswith("bwrap")
224
+ return
225
+
226
+ result = run_tool(
227
+ "shell_command",
228
+ {
229
+ "command": "python - <<'PY'\nimport ctypes, os\nprint(ctypes.CDLL(None).ptrace(0, 0, None, None))\nPY"
230
+ },
231
+ ToolContext(cwd=tmp_path),
232
+ )
233
+
234
+ assert not result.ok or "-1" in result.content
235
+
236
+
237
+ def test_apply_patch_modifies_workdir_file(tmp_path) -> None:
238
+ target = tmp_path / "notes.txt"
239
+ target.write_text("alpha\nbeta\n")
240
+ patch = """*** Begin Patch
241
+ *** Update File: notes.txt
242
+ @@
243
+ -beta
244
+ +ready
245
+ *** End Patch
246
+ """
247
+
248
+ result = run_tool("apply_patch", {"patch": patch}, ToolContext(cwd=tmp_path))
249
+
250
+ assert result.ok
251
+ assert target.read_text() == "alpha\nready\n"
252
+
253
+
254
+ def test_apply_patch_rejects_outside_workdir_file(tmp_path) -> None:
255
+ outside = Path(__file__).resolve().parent / "outside-patch.txt"
256
+ outside.write_text("alpha\n")
257
+ try:
258
+ patch = f"""*** Begin Patch
259
+ *** Update File: {outside}
260
+ @@
261
+ -alpha
262
+ +beta
263
+ *** End Patch
264
+ """
265
+
266
+ result = run_tool("apply_patch", {"patch": patch}, ToolContext(cwd=tmp_path))
267
+
268
+ assert not result.ok
269
+ assert outside.read_text() == "alpha\n"
270
+ finally:
271
+ outside.unlink(missing_ok=True)
272
+
273
+
274
+ def test_apply_patch_uses_internal_subcommand(tmp_path, monkeypatch) -> None:
275
+ calls: list[list[str]] = []
276
+
277
+ def fake_run(self, command, **kwargs):
278
+ calls.append(command)
279
+ from flowent.sandbox import CommandResult
280
+
281
+ return CommandResult(
282
+ command=" ".join(command), exit_code=0, stderr="", stdout="{}"
283
+ )
284
+
285
+ monkeypatch.setattr(SandboxRunner, "run", fake_run)
286
+ patch = """*** Begin Patch
287
+ *** Add File: created.txt
288
+ +hello
289
+ *** End Patch
290
+ """
291
+
292
+ result = run_tool("apply_patch", {"patch": patch}, ToolContext(cwd=tmp_path))
293
+
294
+ assert result.ok
295
+ assert calls
296
+ assert calls[0][1:4] == ["-m", "flowent.cli", "apply-patch"]
297
+
298
+
299
+ def test_apply_patch_reports_patch_error_when_stderr_has_warning(
300
+ tmp_path, monkeypatch
301
+ ) -> None:
302
+ def fake_run(self, command, **kwargs):
303
+ from flowent.sandbox import CommandResult
304
+
305
+ return CommandResult(
306
+ command=" ".join(command),
307
+ exit_code=1,
308
+ stderr="RuntimeWarning: flowent.cli was already imported\n",
309
+ stdout='{"error": "Patch context was not found."}\n',
310
+ )
311
+
312
+ monkeypatch.setattr(SandboxRunner, "run", fake_run)
313
+ patch = """*** Begin Patch
314
+ *** Update File: notes.txt
315
+ @@
316
+ -missing
317
+ +ready
318
+ *** End Patch
319
+ """
320
+
321
+ result = run_tool("apply_patch", {"patch": patch}, ToolContext(cwd=tmp_path))
322
+
323
+ assert not result.ok
324
+ assert result.content == "Patch context was not found."
325
+
326
+
327
+ def test_web_search_result_enters_tool_output(tmp_path) -> None:
328
+ def fake_search(query: str):
329
+ return [{"title": "Result", "url": "https://example.test", "snippet": query}]
330
+
331
+ result = run_tool(
332
+ "web_search",
333
+ {"query": "release checklist"},
334
+ ToolContext(cwd=tmp_path, web_searcher=fake_search),
335
+ )
336
+
337
+ assert result.ok
338
+ assert "https://example.test" in result.content
339
+
340
+
341
+ def test_agent_continues_until_final_text_after_multiple_tool_rounds(
342
+ tmp_path, monkeypatch
343
+ ) -> None:
344
+ monkeypatch.setenv("FLOWENT_DATA_DIR", str(tmp_path / "data"))
345
+ workdir = tmp_path / "workdir"
346
+ workdir.mkdir()
347
+ (workdir / "notes.txt").write_text("Launch notes")
348
+ monkeypatch.chdir(workdir)
349
+ captured_requests: list[dict[str, object]] = []
350
+
351
+ async def fake_completion(**request: object) -> object:
352
+ captured_requests.append(request)
353
+
354
+ async def chunks() -> object:
355
+ if len(captured_requests) == 1:
356
+ yield tool_call_chunk("list_dir", {"path": "."}, call_id="call-list")
357
+ elif len(captured_requests) == 2:
358
+ yield tool_call_chunk(
359
+ "read_file", {"path": "notes.txt"}, call_id="call-read"
360
+ )
361
+ else:
362
+ yield text_chunk("The notes are ready.")
363
+
364
+ return chunks()
365
+
366
+ client = TestClient(
367
+ create_app(serve_frontend=False, chat_completion=fake_completion)
368
+ )
369
+ configure_provider(client)
370
+
371
+ response = client.post(
372
+ "/api/workspace/respond",
373
+ json={"content": "Inspect the workspace."},
374
+ )
375
+
376
+ assert response.status_code == 200
377
+ events = stream_events(response.text)
378
+ assert [event["event"] for event in events] == [
379
+ "start",
380
+ "output_start",
381
+ "tool_start",
382
+ "tool_done",
383
+ "output_start",
384
+ "tool_start",
385
+ "tool_done",
386
+ "output_start",
387
+ "delta",
388
+ "done",
389
+ ]
390
+ assert len(captured_requests) == 3
391
+ assert captured_requests[2]["messages"][-1] == {
392
+ "role": "tool",
393
+ "tool_call_id": "call-read",
394
+ "content": "Launch notes",
395
+ }
396
+ assert events[-1]["data"]["message"]["content"] == "The notes are ready."
397
+
398
+
399
+ def test_agent_finishes_without_tools(tmp_path, monkeypatch) -> None:
400
+ monkeypatch.setenv("FLOWENT_DATA_DIR", str(tmp_path / "data"))
401
+ monkeypatch.chdir(tmp_path)
402
+ captured_requests: list[dict[str, object]] = []
403
+
404
+ async def fake_completion(**request: object) -> object:
405
+ captured_requests.append(request)
406
+
407
+ async def chunks() -> object:
408
+ yield text_chunk("Direct answer.")
409
+
410
+ return chunks()
411
+
412
+ client = TestClient(
413
+ create_app(serve_frontend=False, chat_completion=fake_completion)
414
+ )
415
+ configure_provider(client)
416
+
417
+ response = client.post(
418
+ "/api/workspace/respond",
419
+ json={"content": "Answer directly."},
420
+ )
421
+
422
+ assert response.status_code == 200
423
+ events = stream_events(response.text)
424
+ assert [event["event"] for event in events] == [
425
+ "start",
426
+ "output_start",
427
+ "delta",
428
+ "done",
429
+ ]
430
+ assert len(captured_requests) == 1
431
+ assert events[-1]["data"]["message"]["content"] == "Direct answer."
432
+
433
+
434
+ def test_tool_failure_is_reported_and_agent_continues(tmp_path, monkeypatch) -> None:
435
+ monkeypatch.setenv("FLOWENT_DATA_DIR", str(tmp_path / "data"))
436
+ monkeypatch.chdir(tmp_path)
437
+ captured_requests: list[dict[str, object]] = []
438
+
439
+ async def fake_completion(**request: object) -> object:
440
+ captured_requests.append(request)
441
+
442
+ async def chunks() -> object:
443
+ if len(captured_requests) == 1:
444
+ yield tool_call_chunk("read_file", {"path": "missing.txt"})
445
+ else:
446
+ yield text_chunk("I could not read it.")
447
+
448
+ return chunks()
449
+
450
+ client = TestClient(
451
+ create_app(serve_frontend=False, chat_completion=fake_completion)
452
+ )
453
+ configure_provider(client)
454
+
455
+ response = client.post(
456
+ "/api/workspace/respond",
457
+ json={"content": "Read it."},
458
+ )
459
+
460
+ events = stream_events(response.text)
461
+ assert "tool_error" in [event["event"] for event in events]
462
+ assert len(captured_requests) == 2
463
+ assert captured_requests[1]["messages"][-1]["role"] == "tool"
464
+ assert captured_requests[1]["messages"][-1]["tool_call_id"] == "call-1"
465
+ assert "missing.txt" in captured_requests[1]["messages"][-1]["content"]
466
+ assert events[-1]["data"]["message"]["content"] == "I could not read it."
467
+
468
+
469
+ def test_update_plan_outputs_plan_state(tmp_path) -> None:
470
+ result = run_tool(
471
+ "update_plan",
472
+ {"items": [{"step": "Read files", "status": "completed"}]},
473
+ ToolContext(cwd=tmp_path),
474
+ )
475
+
476
+ assert result.ok
477
+ assert "Read files" in result.content
@@ -0,0 +1,12 @@
1
+ from fastapi.testclient import TestClient
2
+
3
+ from flowent.main import create_app
4
+
5
+
6
+ def test_health_endpoint() -> None:
7
+ client = TestClient(create_app(serve_frontend=False))
8
+
9
+ response = client.get("/api/health")
10
+
11
+ assert response.status_code == 200
12
+ assert response.json() == {"status": "ok"}
@@ -0,0 +1,113 @@
1
+ import pytest
2
+
3
+ from flowent.llm import (
4
+ ChatMessage,
5
+ ProviderConnection,
6
+ ProviderFormat,
7
+ build_litellm_request,
8
+ complete_chat,
9
+ stream_chat,
10
+ )
11
+
12
+
13
+ def test_supported_provider_formats_match_product_choices() -> None:
14
+ assert [provider.value for provider in ProviderFormat] == [
15
+ "openai",
16
+ "openai_responses",
17
+ "anthropic",
18
+ "gemini",
19
+ ]
20
+
21
+
22
+ def test_build_litellm_request_maps_provider_connection_to_completion_args() -> None:
23
+ connection = ProviderConnection(
24
+ name="Primary",
25
+ provider=ProviderFormat.ANTHROPIC,
26
+ model="claude-sonnet-4-5",
27
+ secret_reference="connection-primary",
28
+ base_url="https://example.test/v1",
29
+ )
30
+ messages = [
31
+ ChatMessage(role="system", content="Keep answers direct."),
32
+ ChatMessage(role="user", content="Draft a launch checklist."),
33
+ ]
34
+
35
+ request = build_litellm_request(connection, messages)
36
+
37
+ assert request == {
38
+ "api_base": "https://example.test/v1",
39
+ "api_key": "connection-primary",
40
+ "messages": [
41
+ {"role": "system", "content": "Keep answers direct."},
42
+ {"role": "user", "content": "Draft a launch checklist."},
43
+ ],
44
+ "model": "anthropic/claude-sonnet-4-5",
45
+ }
46
+
47
+
48
+ @pytest.mark.anyio
49
+ async def test_complete_chat_uses_injected_litellm_completion() -> None:
50
+ captured_request: dict[str, object] = {}
51
+
52
+ async def fake_completion(**request: object) -> dict[str, object]:
53
+ captured_request.update(request)
54
+ return {
55
+ "choices": [
56
+ {
57
+ "message": {
58
+ "content": "Here is the checklist.",
59
+ "role": "assistant",
60
+ },
61
+ }
62
+ ]
63
+ }
64
+
65
+ connection = ProviderConnection(
66
+ name="Responses",
67
+ provider=ProviderFormat.OPENAI_RESPONSES,
68
+ model="gpt-5.1",
69
+ secret_reference="connection-responses",
70
+ )
71
+
72
+ answer = await complete_chat(
73
+ connection,
74
+ [ChatMessage(role="user", content="Create a checklist.")],
75
+ completion=fake_completion,
76
+ )
77
+
78
+ assert captured_request["model"] == "openai/gpt-5.1"
79
+ assert answer == ChatMessage(role="assistant", content="Here is the checklist.")
80
+
81
+
82
+ @pytest.mark.anyio
83
+ async def test_stream_chat_uses_litellm_streaming() -> None:
84
+ captured_request: dict[str, object] = {}
85
+
86
+ async def fake_completion(**request: object) -> object:
87
+ captured_request.update(request)
88
+
89
+ async def chunks() -> object:
90
+ yield {"choices": [{"delta": {"content": "Here is "}}]}
91
+ yield {"choices": [{"delta": {"content": "the checklist."}}]}
92
+
93
+ return chunks()
94
+
95
+ connection = ProviderConnection(
96
+ name="Responses",
97
+ provider=ProviderFormat.OPENAI_RESPONSES,
98
+ model="gpt-5.1",
99
+ secret_reference="connection-responses",
100
+ )
101
+
102
+ chunks = [
103
+ chunk
104
+ async for chunk in stream_chat(
105
+ connection,
106
+ [ChatMessage(role="user", content="Create a checklist.")],
107
+ completion=fake_completion,
108
+ )
109
+ ]
110
+
111
+ assert captured_request["stream"] is True
112
+ assert captured_request["model"] == "openai/gpt-5.1"
113
+ assert chunks == ["Here is ", "the checklist."]