flowent 0.0.1 → 0.0.5

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (484) hide show
  1. package/README.md +20 -9
  2. package/backend/.python-version +1 -0
  3. package/backend/README.md +74 -0
  4. package/backend/pyproject.toml +58 -0
  5. package/backend/src/flowent/__init__.py +3 -0
  6. package/backend/src/flowent/__pycache__/__init__.cpython-313.pyc +0 -0
  7. package/backend/src/flowent/__pycache__/_version.cpython-313.pyc +0 -0
  8. package/backend/src/flowent/__pycache__/access.cpython-313.pyc +0 -0
  9. package/backend/src/flowent/__pycache__/agent.cpython-313.pyc +0 -0
  10. package/backend/src/flowent/__pycache__/assistant_commands.cpython-313.pyc +0 -0
  11. package/backend/src/flowent/__pycache__/cli.cpython-313.pyc +0 -0
  12. package/backend/src/flowent/__pycache__/config.cpython-313.pyc +0 -0
  13. package/backend/src/flowent/__pycache__/events.cpython-313.pyc +0 -0
  14. package/backend/src/flowent/__pycache__/graph_runtime.cpython-313.pyc +0 -0
  15. package/backend/src/flowent/__pycache__/graph_service.cpython-313.pyc +0 -0
  16. package/backend/src/flowent/__pycache__/image_assets.cpython-313.pyc +0 -0
  17. package/backend/src/flowent/__pycache__/logging.cpython-313.pyc +0 -0
  18. package/backend/src/flowent/__pycache__/main.cpython-313.pyc +0 -0
  19. package/backend/src/flowent/__pycache__/mcp_service.cpython-313.pyc +0 -0
  20. package/backend/src/flowent/__pycache__/model_metadata.cpython-313.pyc +0 -0
  21. package/backend/src/flowent/__pycache__/network.cpython-313.pyc +0 -0
  22. package/backend/src/flowent/__pycache__/observability_service.cpython-313.pyc +0 -0
  23. package/backend/src/flowent/__pycache__/registry.cpython-313.pyc +0 -0
  24. package/backend/src/flowent/__pycache__/role_management.cpython-313.pyc +0 -0
  25. package/backend/src/flowent/__pycache__/runtime.cpython-313.pyc +0 -0
  26. package/backend/src/flowent/__pycache__/sandbox.cpython-313.pyc +0 -0
  27. package/backend/src/flowent/__pycache__/security.cpython-313.pyc +0 -0
  28. package/backend/src/flowent/__pycache__/settings.cpython-313.pyc +0 -0
  29. package/backend/src/flowent/__pycache__/settings_management.cpython-313.pyc +0 -0
  30. package/backend/src/flowent/__pycache__/state_db.cpython-313.pyc +0 -0
  31. package/backend/src/flowent/__pycache__/workspace_store.cpython-313.pyc +0 -0
  32. package/backend/src/flowent/_version.py +7 -0
  33. package/backend/src/flowent/access.py +247 -0
  34. package/backend/src/flowent/agent.py +3120 -0
  35. package/backend/src/flowent/assistant_commands.py +115 -0
  36. package/backend/src/flowent/channels/__init__.py +3 -0
  37. package/backend/src/flowent/channels/__pycache__/__init__.cpython-313.pyc +0 -0
  38. package/backend/src/flowent/channels/__pycache__/telegram.cpython-313.pyc +0 -0
  39. package/backend/src/flowent/channels/telegram.py +615 -0
  40. package/backend/src/flowent/cli.py +85 -0
  41. package/backend/src/flowent/config.py +14 -0
  42. package/backend/src/flowent/dev.py +3 -0
  43. package/backend/src/flowent/events.py +157 -0
  44. package/backend/src/flowent/graph_runtime.py +60 -0
  45. package/backend/src/flowent/graph_service.py +2508 -0
  46. package/backend/src/flowent/image_assets.py +356 -0
  47. package/backend/src/flowent/logging.py +155 -0
  48. package/backend/src/flowent/main.py +124 -0
  49. package/backend/src/flowent/mcp_service.py +1918 -0
  50. package/backend/src/flowent/model_metadata.py +102 -0
  51. package/backend/src/flowent/models/__init__.py +125 -0
  52. package/backend/src/flowent/models/__pycache__/__init__.cpython-313.pyc +0 -0
  53. package/backend/src/flowent/models/__pycache__/agent.cpython-313.pyc +0 -0
  54. package/backend/src/flowent/models/__pycache__/base.cpython-313.pyc +0 -0
  55. package/backend/src/flowent/models/__pycache__/blueprint.cpython-313.pyc +0 -0
  56. package/backend/src/flowent/models/__pycache__/content.cpython-313.pyc +0 -0
  57. package/backend/src/flowent/models/__pycache__/delta.cpython-313.pyc +0 -0
  58. package/backend/src/flowent/models/__pycache__/event.cpython-313.pyc +0 -0
  59. package/backend/src/flowent/models/__pycache__/graph.cpython-313.pyc +0 -0
  60. package/backend/src/flowent/models/__pycache__/history.cpython-313.pyc +0 -0
  61. package/backend/src/flowent/models/__pycache__/llm.cpython-313.pyc +0 -0
  62. package/backend/src/flowent/models/__pycache__/message.cpython-313.pyc +0 -0
  63. package/backend/src/flowent/models/__pycache__/tab.cpython-313.pyc +0 -0
  64. package/backend/src/flowent/models/__pycache__/todo.cpython-313.pyc +0 -0
  65. package/backend/src/flowent/models/agent.py +34 -0
  66. package/backend/src/flowent/models/base.py +24 -0
  67. package/backend/src/flowent/models/blueprint.py +176 -0
  68. package/backend/src/flowent/models/content.py +164 -0
  69. package/backend/src/flowent/models/delta.py +44 -0
  70. package/backend/src/flowent/models/event.py +51 -0
  71. package/backend/src/flowent/models/graph.py +472 -0
  72. package/backend/src/flowent/models/history.py +272 -0
  73. package/backend/src/flowent/models/llm.py +62 -0
  74. package/backend/src/flowent/models/message.py +33 -0
  75. package/backend/src/flowent/models/tab.py +85 -0
  76. package/backend/src/flowent/models/todo.py +10 -0
  77. package/backend/src/flowent/network.py +146 -0
  78. package/backend/src/flowent/observability_service.py +218 -0
  79. package/backend/src/flowent/prompts/__init__.py +67 -0
  80. package/backend/src/flowent/prompts/__pycache__/__init__.cpython-313.pyc +0 -0
  81. package/backend/src/flowent/prompts/__pycache__/common.cpython-313.pyc +0 -0
  82. package/backend/src/flowent/prompts/__pycache__/steward.cpython-313.pyc +0 -0
  83. package/backend/src/flowent/prompts/common.py +250 -0
  84. package/backend/src/flowent/prompts/steward.py +64 -0
  85. package/backend/src/flowent/providers/__init__.py +23 -0
  86. package/backend/src/flowent/providers/__pycache__/__init__.cpython-313.pyc +0 -0
  87. package/backend/src/flowent/providers/__pycache__/anthropic.cpython-313.pyc +0 -0
  88. package/backend/src/flowent/providers/__pycache__/base_url.cpython-313.pyc +0 -0
  89. package/backend/src/flowent/providers/__pycache__/configuration.cpython-313.pyc +0 -0
  90. package/backend/src/flowent/providers/__pycache__/content.cpython-313.pyc +0 -0
  91. package/backend/src/flowent/providers/__pycache__/errors.cpython-313.pyc +0 -0
  92. package/backend/src/flowent/providers/__pycache__/gateway.cpython-313.pyc +0 -0
  93. package/backend/src/flowent/providers/__pycache__/headers.cpython-313.pyc +0 -0
  94. package/backend/src/flowent/providers/__pycache__/management.cpython-313.pyc +0 -0
  95. package/backend/src/flowent/providers/__pycache__/openai.cpython-313.pyc +0 -0
  96. package/backend/src/flowent/providers/__pycache__/openai_responses.cpython-313.pyc +0 -0
  97. package/backend/src/flowent/providers/__pycache__/registry.cpython-313.pyc +0 -0
  98. package/backend/src/flowent/providers/__pycache__/sse.cpython-313.pyc +0 -0
  99. package/backend/src/flowent/providers/__pycache__/thinking.cpython-313.pyc +0 -0
  100. package/backend/src/flowent/providers/anthropic.py +468 -0
  101. package/backend/src/flowent/providers/base_url.py +60 -0
  102. package/backend/src/flowent/providers/configuration.py +189 -0
  103. package/backend/src/flowent/providers/content.py +122 -0
  104. package/backend/src/flowent/providers/errors.py +223 -0
  105. package/backend/src/flowent/providers/gateway.py +169 -0
  106. package/backend/src/flowent/providers/gemini.py +447 -0
  107. package/backend/src/flowent/providers/headers.py +20 -0
  108. package/backend/src/flowent/providers/management.py +96 -0
  109. package/backend/src/flowent/providers/ollama.py +293 -0
  110. package/backend/src/flowent/providers/openai.py +422 -0
  111. package/backend/src/flowent/providers/openai_responses.py +655 -0
  112. package/backend/src/flowent/providers/registry.py +144 -0
  113. package/backend/src/flowent/providers/sse.py +31 -0
  114. package/backend/src/flowent/providers/thinking.py +79 -0
  115. package/backend/src/flowent/registry.py +73 -0
  116. package/backend/src/flowent/role_management.py +267 -0
  117. package/backend/src/flowent/routes/__init__.py +28 -0
  118. package/backend/src/flowent/routes/__pycache__/__init__.cpython-313.pyc +0 -0
  119. package/backend/src/flowent/routes/__pycache__/access.cpython-313.pyc +0 -0
  120. package/backend/src/flowent/routes/__pycache__/assistant.cpython-313.pyc +0 -0
  121. package/backend/src/flowent/routes/__pycache__/image_assets.cpython-313.pyc +0 -0
  122. package/backend/src/flowent/routes/__pycache__/mcp.cpython-313.pyc +0 -0
  123. package/backend/src/flowent/routes/__pycache__/meta.cpython-313.pyc +0 -0
  124. package/backend/src/flowent/routes/__pycache__/nodes.cpython-313.pyc +0 -0
  125. package/backend/src/flowent/routes/__pycache__/prompts.cpython-313.pyc +0 -0
  126. package/backend/src/flowent/routes/__pycache__/providers_route.cpython-313.pyc +0 -0
  127. package/backend/src/flowent/routes/__pycache__/roles.cpython-313.pyc +0 -0
  128. package/backend/src/flowent/routes/__pycache__/settings.cpython-313.pyc +0 -0
  129. package/backend/src/flowent/routes/__pycache__/tabs.cpython-313.pyc +0 -0
  130. package/backend/src/flowent/routes/__pycache__/ws.cpython-313.pyc +0 -0
  131. package/backend/src/flowent/routes/access.py +48 -0
  132. package/backend/src/flowent/routes/assistant.py +155 -0
  133. package/backend/src/flowent/routes/image_assets.py +33 -0
  134. package/backend/src/flowent/routes/mcp.py +125 -0
  135. package/backend/src/flowent/routes/meta.py +28 -0
  136. package/backend/src/flowent/routes/nodes.py +413 -0
  137. package/backend/src/flowent/routes/prompts.py +46 -0
  138. package/backend/src/flowent/routes/providers_route.py +365 -0
  139. package/backend/src/flowent/routes/roles.py +207 -0
  140. package/backend/src/flowent/routes/settings.py +328 -0
  141. package/backend/src/flowent/routes/tabs.py +310 -0
  142. package/backend/src/flowent/routes/ws.py +33 -0
  143. package/backend/src/flowent/runtime.py +165 -0
  144. package/backend/src/flowent/sandbox.py +45 -0
  145. package/backend/src/flowent/security.py +57 -0
  146. package/backend/src/flowent/settings.py +2518 -0
  147. package/backend/src/flowent/settings_management.py +298 -0
  148. package/backend/src/flowent/state_db.py +120 -0
  149. package/backend/src/flowent/static/assets/AssistantPage-VBohhz4d.js +1 -0
  150. package/backend/src/flowent/static/assets/ChannelsPage-CIydPZA_.js +1 -0
  151. package/backend/src/flowent/static/assets/McpPage-CHPm2TPY.js +7 -0
  152. package/backend/src/flowent/static/assets/PageScaffold-DteOA8V7.js +1 -0
  153. package/backend/src/flowent/static/assets/PromptsPage-CSmJ3sZg.js +1 -0
  154. package/backend/src/flowent/static/assets/ProvidersPage-sl2jeG4e.js +3 -0
  155. package/backend/src/flowent/static/assets/RolesPage-DCe7W6Km.js +1 -0
  156. package/backend/src/flowent/static/assets/SettingsPage-Bix9e63E.js +3 -0
  157. package/backend/src/flowent/static/assets/ToolsPage-favNkj5C.js +1 -0
  158. package/backend/src/flowent/static/assets/WorkspaceCommandDialog-DRS6wiD6.js +1 -0
  159. package/backend/src/flowent/static/assets/WorkspacePage-KuaDjt_D.js +3 -0
  160. package/backend/src/flowent/static/assets/WorkspacePanels-BZxBw8M5.js +1 -0
  161. package/backend/src/flowent/static/assets/alert-dialog-DIBUCmqM.js +1 -0
  162. package/backend/src/flowent/static/assets/datetime-eJqd0V2S.js +1 -0
  163. package/backend/src/flowent/static/assets/dialog-BOvHIBrg.js +1 -0
  164. package/backend/src/flowent/static/assets/elk-worker.min-C9JGDOE-.js +6312 -0
  165. package/backend/src/flowent/static/assets/graph-vendor-CHpVij2M.css +1 -0
  166. package/backend/src/flowent/static/assets/graph-vendor-DRq_-6fV.js +7 -0
  167. package/backend/src/flowent/static/assets/index-Biio-CoI.js +10 -0
  168. package/backend/src/flowent/static/assets/index-CmQvO7sl.css +1 -0
  169. package/backend/src/flowent/static/assets/layout.worker-jMHqAFbP.js +24 -0
  170. package/backend/src/flowent/static/assets/markdown-vendor-C9RtvaJh.js +29 -0
  171. package/backend/src/flowent/static/assets/modelParams-DcEhGnu0.js +1 -0
  172. package/backend/src/flowent/static/assets/react-vendor-mEs_JJxa.js +9 -0
  173. package/backend/src/flowent/static/assets/roles-BbIEIMeG.js +1 -0
  174. package/backend/src/flowent/static/assets/rolldown-runtime-BYbx6iT9.js +1 -0
  175. package/backend/src/flowent/static/assets/select-D9SwnlXF.js +1 -0
  176. package/backend/src/flowent/static/assets/surface-Bzr1FRG4.js +1 -0
  177. package/backend/src/flowent/static/assets/triState-DgLlKdRR.js +1 -0
  178. package/backend/src/flowent/static/assets/ui-vendor-UazN8rcv.js +51 -0
  179. package/backend/src/flowent/static/index.html +35 -0
  180. package/backend/src/flowent/tools/__init__.py +275 -0
  181. package/backend/src/flowent/tools/__pycache__/__init__.cpython-313.pyc +0 -0
  182. package/backend/src/flowent/tools/__pycache__/connect.cpython-313.pyc +0 -0
  183. package/backend/src/flowent/tools/__pycache__/contacts.cpython-313.pyc +0 -0
  184. package/backend/src/flowent/tools/__pycache__/create_agent.cpython-313.pyc +0 -0
  185. package/backend/src/flowent/tools/__pycache__/create_tab.cpython-313.pyc +0 -0
  186. package/backend/src/flowent/tools/__pycache__/delete_tab.cpython-313.pyc +0 -0
  187. package/backend/src/flowent/tools/__pycache__/edit.cpython-313.pyc +0 -0
  188. package/backend/src/flowent/tools/__pycache__/exec.cpython-313.pyc +0 -0
  189. package/backend/src/flowent/tools/__pycache__/fetch.cpython-313.pyc +0 -0
  190. package/backend/src/flowent/tools/__pycache__/idle.cpython-313.pyc +0 -0
  191. package/backend/src/flowent/tools/__pycache__/list_roles.cpython-313.pyc +0 -0
  192. package/backend/src/flowent/tools/__pycache__/list_tabs.cpython-313.pyc +0 -0
  193. package/backend/src/flowent/tools/__pycache__/list_tools.cpython-313.pyc +0 -0
  194. package/backend/src/flowent/tools/__pycache__/manage_prompts.cpython-313.pyc +0 -0
  195. package/backend/src/flowent/tools/__pycache__/manage_providers.cpython-313.pyc +0 -0
  196. package/backend/src/flowent/tools/__pycache__/manage_roles.cpython-313.pyc +0 -0
  197. package/backend/src/flowent/tools/__pycache__/manage_settings.cpython-313.pyc +0 -0
  198. package/backend/src/flowent/tools/__pycache__/mcp.cpython-313.pyc +0 -0
  199. package/backend/src/flowent/tools/__pycache__/read.cpython-313.pyc +0 -0
  200. package/backend/src/flowent/tools/__pycache__/send.cpython-313.pyc +0 -0
  201. package/backend/src/flowent/tools/__pycache__/set_permissions.cpython-313.pyc +0 -0
  202. package/backend/src/flowent/tools/__pycache__/sleep.cpython-313.pyc +0 -0
  203. package/backend/src/flowent/tools/__pycache__/todo.cpython-313.pyc +0 -0
  204. package/backend/src/flowent/tools/connect.py +100 -0
  205. package/backend/src/flowent/tools/contacts.py +22 -0
  206. package/backend/src/flowent/tools/create_agent.py +191 -0
  207. package/backend/src/flowent/tools/create_tab.py +61 -0
  208. package/backend/src/flowent/tools/delete_tab.py +39 -0
  209. package/backend/src/flowent/tools/edit.py +142 -0
  210. package/backend/src/flowent/tools/exec.py +118 -0
  211. package/backend/src/flowent/tools/fetch.py +85 -0
  212. package/backend/src/flowent/tools/idle.py +27 -0
  213. package/backend/src/flowent/tools/list_roles.py +75 -0
  214. package/backend/src/flowent/tools/list_tabs.py +100 -0
  215. package/backend/src/flowent/tools/list_tools.py +28 -0
  216. package/backend/src/flowent/tools/manage_prompts.py +102 -0
  217. package/backend/src/flowent/tools/manage_providers.py +220 -0
  218. package/backend/src/flowent/tools/manage_roles.py +275 -0
  219. package/backend/src/flowent/tools/manage_settings.py +364 -0
  220. package/backend/src/flowent/tools/mcp.py +199 -0
  221. package/backend/src/flowent/tools/read.py +152 -0
  222. package/backend/src/flowent/tools/send.py +68 -0
  223. package/backend/src/flowent/tools/set_permissions.py +99 -0
  224. package/backend/src/flowent/tools/sleep.py +41 -0
  225. package/backend/src/flowent/tools/todo.py +51 -0
  226. package/backend/src/flowent/workspace_store.py +479 -0
  227. package/backend/tests/__init__.py +0 -0
  228. package/backend/tests/__pycache__/__init__.cpython-313.pyc +0 -0
  229. package/backend/tests/__pycache__/conftest.cpython-313-pytest-9.0.3.pyc +0 -0
  230. package/backend/tests/conftest.py +6 -0
  231. package/backend/tests/integration/api/__pycache__/conftest.cpython-313-pytest-9.0.3.pyc +0 -0
  232. package/backend/tests/integration/api/__pycache__/test_access_api.cpython-313-pytest-9.0.3.pyc +0 -0
  233. package/backend/tests/integration/api/__pycache__/test_assistant_api.cpython-313-pytest-9.0.3.pyc +0 -0
  234. package/backend/tests/integration/api/__pycache__/test_frontend_mounting.cpython-313-pytest-9.0.3.pyc +0 -0
  235. package/backend/tests/integration/api/__pycache__/test_mcp_api.cpython-313-pytest-9.0.3.pyc +0 -0
  236. package/backend/tests/integration/api/__pycache__/test_meta_api.cpython-313-pytest-9.0.3.pyc +0 -0
  237. package/backend/tests/integration/api/__pycache__/test_nodes_api.cpython-313-pytest-9.0.3.pyc +0 -0
  238. package/backend/tests/integration/api/__pycache__/test_prompts_api.cpython-313-pytest-9.0.3.pyc +0 -0
  239. package/backend/tests/integration/api/__pycache__/test_roles_api.cpython-313-pytest-9.0.3.pyc +0 -0
  240. package/backend/tests/integration/api/__pycache__/test_tabs_api.cpython-313-pytest-9.0.3.pyc +0 -0
  241. package/backend/tests/integration/api/conftest.py +29 -0
  242. package/backend/tests/integration/api/test_access_api.py +182 -0
  243. package/backend/tests/integration/api/test_assistant_api.py +354 -0
  244. package/backend/tests/integration/api/test_frontend_mounting.py +61 -0
  245. package/backend/tests/integration/api/test_mcp_api.py +116 -0
  246. package/backend/tests/integration/api/test_meta_api.py +33 -0
  247. package/backend/tests/integration/api/test_nodes_api.py +722 -0
  248. package/backend/tests/integration/api/test_prompts_api.py +47 -0
  249. package/backend/tests/integration/api/test_roles_api.py +228 -0
  250. package/backend/tests/integration/api/test_tabs_api.py +802 -0
  251. package/backend/tests/unit/__pycache__/test_access.cpython-313-pytest-9.0.3.pyc +0 -0
  252. package/backend/tests/unit/__pycache__/test_cli.cpython-313-pytest-9.0.3.pyc +0 -0
  253. package/backend/tests/unit/__pycache__/test_graph_runtime.cpython-313-pytest-9.0.3.pyc +0 -0
  254. package/backend/tests/unit/__pycache__/test_network.cpython-313-pytest-9.0.3.pyc +0 -0
  255. package/backend/tests/unit/__pycache__/test_state_sqlite_storage.cpython-313-pytest-9.0.3.pyc +0 -0
  256. package/backend/tests/unit/__pycache__/test_workspace_store.cpython-313-pytest-9.0.3.pyc +0 -0
  257. package/backend/tests/unit/agent/__pycache__/test_agent_public_api.cpython-313-pytest-9.0.3.pyc +0 -0
  258. package/backend/tests/unit/agent/__pycache__/test_agent_runtime.cpython-313-pytest-9.0.3.pyc +0 -0
  259. package/backend/tests/unit/agent/test_agent_public_api.py +837 -0
  260. package/backend/tests/unit/agent/test_agent_runtime.py +2942 -0
  261. package/backend/tests/unit/channels/__pycache__/test_telegram_channel.cpython-313-pytest-9.0.3.pyc +0 -0
  262. package/backend/tests/unit/channels/test_telegram_channel.py +552 -0
  263. package/backend/tests/unit/logging/__pycache__/test_logging.cpython-313-pytest-9.0.3.pyc +0 -0
  264. package/backend/tests/unit/logging/test_logging.py +132 -0
  265. package/backend/tests/unit/prompts/__pycache__/test_prompts.cpython-313-pytest-9.0.3.pyc +0 -0
  266. package/backend/tests/unit/prompts/test_prompts.py +570 -0
  267. package/backend/tests/unit/providers/__pycache__/test_anthropic_provider.cpython-313-pytest-9.0.3.pyc +0 -0
  268. package/backend/tests/unit/providers/__pycache__/test_errors.cpython-313-pytest-9.0.3.pyc +0 -0
  269. package/backend/tests/unit/providers/__pycache__/test_extract_delta_parts.cpython-313-pytest-9.0.3.pyc +0 -0
  270. package/backend/tests/unit/providers/__pycache__/test_openai_provider.cpython-313-pytest-9.0.3.pyc +0 -0
  271. package/backend/tests/unit/providers/__pycache__/test_openai_responses.cpython-313-pytest-9.0.3.pyc +0 -0
  272. package/backend/tests/unit/providers/__pycache__/test_provider_gateway.cpython-313-pytest-9.0.3.pyc +0 -0
  273. package/backend/tests/unit/providers/__pycache__/test_think_tag_parser.cpython-313-pytest-9.0.3.pyc +0 -0
  274. package/backend/tests/unit/providers/test_anthropic_provider.py +185 -0
  275. package/backend/tests/unit/providers/test_errors.py +68 -0
  276. package/backend/tests/unit/providers/test_extract_delta_parts.py +22 -0
  277. package/backend/tests/unit/providers/test_openai_provider.py +139 -0
  278. package/backend/tests/unit/providers/test_openai_responses.py +402 -0
  279. package/backend/tests/unit/providers/test_provider_gateway.py +359 -0
  280. package/backend/tests/unit/providers/test_think_tag_parser.py +36 -0
  281. package/backend/tests/unit/routes/__pycache__/test_prompts_routes.cpython-313-pytest-9.0.3.pyc +0 -0
  282. package/backend/tests/unit/routes/__pycache__/test_providers_route.cpython-313-pytest-9.0.3.pyc +0 -0
  283. package/backend/tests/unit/routes/__pycache__/test_roles_routes.cpython-313-pytest-9.0.3.pyc +0 -0
  284. package/backend/tests/unit/routes/__pycache__/test_settings_routes.cpython-313-pytest-9.0.3.pyc +0 -0
  285. package/backend/tests/unit/routes/test_prompts_routes.py +104 -0
  286. package/backend/tests/unit/routes/test_providers_route.py +370 -0
  287. package/backend/tests/unit/routes/test_roles_routes.py +535 -0
  288. package/backend/tests/unit/routes/test_settings_routes.py +1142 -0
  289. package/backend/tests/unit/runtime/__pycache__/test_bootstrap_runtime.cpython-313-pytest-9.0.3.pyc +0 -0
  290. package/backend/tests/unit/runtime/test_bootstrap_runtime.py +1002 -0
  291. package/backend/tests/unit/sandbox/__pycache__/test_sandbox_tools.cpython-313-pytest-9.0.3.pyc +0 -0
  292. package/backend/tests/unit/sandbox/test_sandbox_tools.py +78 -0
  293. package/backend/tests/unit/security/__pycache__/test_security.cpython-313-pytest-9.0.3.pyc +0 -0
  294. package/backend/tests/unit/security/test_security.py +124 -0
  295. package/backend/tests/unit/settings/__pycache__/test_settings_roles.cpython-313-pytest-9.0.3.pyc +0 -0
  296. package/backend/tests/unit/settings/test_settings_roles.py +751 -0
  297. package/backend/tests/unit/test_access.py +45 -0
  298. package/backend/tests/unit/test_cli.py +124 -0
  299. package/backend/tests/unit/test_graph_runtime.py +72 -0
  300. package/backend/tests/unit/test_network.py +51 -0
  301. package/backend/tests/unit/test_state_sqlite_storage.py +159 -0
  302. package/backend/tests/unit/test_workspace_store.py +231 -0
  303. package/backend/tests/unit/tools/__pycache__/test_connect_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  304. package/backend/tests/unit/tools/__pycache__/test_create_agent_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  305. package/backend/tests/unit/tools/__pycache__/test_delete_tab_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  306. package/backend/tests/unit/tools/__pycache__/test_edit_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  307. package/backend/tests/unit/tools/__pycache__/test_exec_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  308. package/backend/tests/unit/tools/__pycache__/test_fetch_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  309. package/backend/tests/unit/tools/__pycache__/test_manage_prompts_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  310. package/backend/tests/unit/tools/__pycache__/test_manage_providers_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  311. package/backend/tests/unit/tools/__pycache__/test_manage_roles_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  312. package/backend/tests/unit/tools/__pycache__/test_manage_settings_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  313. package/backend/tests/unit/tools/__pycache__/test_read_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  314. package/backend/tests/unit/tools/__pycache__/test_set_permissions_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  315. package/backend/tests/unit/tools/__pycache__/test_todo_tool.cpython-313-pytest-9.0.3.pyc +0 -0
  316. package/backend/tests/unit/tools/__pycache__/test_tool_registry.cpython-313-pytest-9.0.3.pyc +0 -0
  317. package/backend/tests/unit/tools/test_connect_tool.py +228 -0
  318. package/backend/tests/unit/tools/test_create_agent_tool.py +436 -0
  319. package/backend/tests/unit/tools/test_delete_tab_tool.py +116 -0
  320. package/backend/tests/unit/tools/test_edit_tool.py +115 -0
  321. package/backend/tests/unit/tools/test_exec_tool.py +81 -0
  322. package/backend/tests/unit/tools/test_fetch_tool.py +65 -0
  323. package/backend/tests/unit/tools/test_manage_prompts_tool.py +117 -0
  324. package/backend/tests/unit/tools/test_manage_providers_tool.py +460 -0
  325. package/backend/tests/unit/tools/test_manage_roles_tool.py +411 -0
  326. package/backend/tests/unit/tools/test_manage_settings_tool.py +611 -0
  327. package/backend/tests/unit/tools/test_read_tool.py +33 -0
  328. package/backend/tests/unit/tools/test_set_permissions_tool.py +595 -0
  329. package/backend/tests/unit/tools/test_todo_tool.py +37 -0
  330. package/backend/tests/unit/tools/test_tool_registry.py +194 -0
  331. package/backend/uv.lock +1144 -0
  332. package/bin/flowent.mjs +62 -36
  333. package/dist/frontend/assets/AssistantPage-VBohhz4d.js +1 -0
  334. package/dist/frontend/assets/ChannelsPage-CIydPZA_.js +1 -0
  335. package/dist/frontend/assets/McpPage-CHPm2TPY.js +7 -0
  336. package/dist/frontend/assets/PageScaffold-DteOA8V7.js +1 -0
  337. package/dist/frontend/assets/PromptsPage-CSmJ3sZg.js +1 -0
  338. package/dist/frontend/assets/ProvidersPage-sl2jeG4e.js +3 -0
  339. package/dist/frontend/assets/RolesPage-DCe7W6Km.js +1 -0
  340. package/dist/frontend/assets/SettingsPage-Bix9e63E.js +3 -0
  341. package/dist/frontend/assets/ToolsPage-favNkj5C.js +1 -0
  342. package/dist/frontend/assets/WorkspaceCommandDialog-DRS6wiD6.js +1 -0
  343. package/dist/frontend/assets/WorkspacePage-KuaDjt_D.js +3 -0
  344. package/dist/frontend/assets/WorkspacePanels-BZxBw8M5.js +1 -0
  345. package/dist/frontend/assets/alert-dialog-DIBUCmqM.js +1 -0
  346. package/dist/frontend/assets/datetime-eJqd0V2S.js +1 -0
  347. package/dist/frontend/assets/dialog-BOvHIBrg.js +1 -0
  348. package/dist/frontend/assets/elk-worker.min-C9JGDOE-.js +6312 -0
  349. package/dist/frontend/assets/graph-vendor-CHpVij2M.css +1 -0
  350. package/dist/frontend/assets/graph-vendor-DRq_-6fV.js +7 -0
  351. package/dist/frontend/assets/index-Biio-CoI.js +10 -0
  352. package/dist/frontend/assets/index-CmQvO7sl.css +1 -0
  353. package/dist/frontend/assets/layout.worker-jMHqAFbP.js +24 -0
  354. package/dist/frontend/assets/markdown-vendor-C9RtvaJh.js +29 -0
  355. package/dist/frontend/assets/modelParams-DcEhGnu0.js +1 -0
  356. package/dist/frontend/assets/react-vendor-mEs_JJxa.js +9 -0
  357. package/dist/frontend/assets/roles-BbIEIMeG.js +1 -0
  358. package/dist/frontend/assets/rolldown-runtime-BYbx6iT9.js +1 -0
  359. package/dist/frontend/assets/select-D9SwnlXF.js +1 -0
  360. package/dist/frontend/assets/surface-Bzr1FRG4.js +1 -0
  361. package/dist/frontend/assets/triState-DgLlKdRR.js +1 -0
  362. package/dist/frontend/assets/ui-vendor-UazN8rcv.js +51 -0
  363. package/dist/frontend/index.html +35 -0
  364. package/package.json +27 -41
  365. package/dist/.next/BUILD_ID +0 -1
  366. package/dist/.next/app-path-routes-manifest.json +0 -6
  367. package/dist/.next/build-manifest.json +0 -20
  368. package/dist/.next/package.json +0 -1
  369. package/dist/.next/prerender-manifest.json +0 -114
  370. package/dist/.next/required-server-files.json +0 -333
  371. package/dist/.next/routes-manifest.json +0 -69
  372. package/dist/.next/server/app/_global-error/page/app-paths-manifest.json +0 -3
  373. package/dist/.next/server/app/_global-error/page/build-manifest.json +0 -16
  374. package/dist/.next/server/app/_global-error/page/next-font-manifest.json +0 -6
  375. package/dist/.next/server/app/_global-error/page/react-loadable-manifest.json +0 -1
  376. package/dist/.next/server/app/_global-error/page/server-reference-manifest.json +0 -4
  377. package/dist/.next/server/app/_global-error/page.js +0 -9
  378. package/dist/.next/server/app/_global-error/page.js.map +0 -5
  379. package/dist/.next/server/app/_global-error/page.js.nft.json +0 -1
  380. package/dist/.next/server/app/_global-error/page_client-reference-manifest.js +0 -3
  381. package/dist/.next/server/app/_global-error.html +0 -1
  382. package/dist/.next/server/app/_global-error.meta +0 -15
  383. package/dist/.next/server/app/_global-error.rsc +0 -14
  384. package/dist/.next/server/app/_global-error.segments/__PAGE__.segment.rsc +0 -5
  385. package/dist/.next/server/app/_global-error.segments/_full.segment.rsc +0 -14
  386. package/dist/.next/server/app/_global-error.segments/_head.segment.rsc +0 -5
  387. package/dist/.next/server/app/_global-error.segments/_index.segment.rsc +0 -5
  388. package/dist/.next/server/app/_global-error.segments/_tree.segment.rsc +0 -1
  389. package/dist/.next/server/app/_not-found/page/app-paths-manifest.json +0 -3
  390. package/dist/.next/server/app/_not-found/page/build-manifest.json +0 -16
  391. package/dist/.next/server/app/_not-found/page/next-font-manifest.json +0 -10
  392. package/dist/.next/server/app/_not-found/page/react-loadable-manifest.json +0 -1
  393. package/dist/.next/server/app/_not-found/page/server-reference-manifest.json +0 -4
  394. package/dist/.next/server/app/_not-found/page.js +0 -13
  395. package/dist/.next/server/app/_not-found/page.js.map +0 -5
  396. package/dist/.next/server/app/_not-found/page.js.nft.json +0 -1
  397. package/dist/.next/server/app/_not-found/page_client-reference-manifest.js +0 -3
  398. package/dist/.next/server/app/_not-found.html +0 -1
  399. package/dist/.next/server/app/_not-found.meta +0 -16
  400. package/dist/.next/server/app/_not-found.rsc +0 -16
  401. package/dist/.next/server/app/_not-found.segments/_full.segment.rsc +0 -16
  402. package/dist/.next/server/app/_not-found.segments/_head.segment.rsc +0 -6
  403. package/dist/.next/server/app/_not-found.segments/_index.segment.rsc +0 -5
  404. package/dist/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +0 -5
  405. package/dist/.next/server/app/_not-found.segments/_not-found.segment.rsc +0 -5
  406. package/dist/.next/server/app/_not-found.segments/_tree.segment.rsc +0 -2
  407. package/dist/.next/server/app/icon.svg/route/app-paths-manifest.json +0 -3
  408. package/dist/.next/server/app/icon.svg/route/build-manifest.json +0 -9
  409. package/dist/.next/server/app/icon.svg/route.js +0 -6
  410. package/dist/.next/server/app/icon.svg/route.js.map +0 -5
  411. package/dist/.next/server/app/icon.svg/route.js.nft.json +0 -1
  412. package/dist/.next/server/app/icon.svg.meta +0 -1
  413. package/dist/.next/server/app/index.html +0 -1
  414. package/dist/.next/server/app/index.meta +0 -14
  415. package/dist/.next/server/app/index.rsc +0 -15
  416. package/dist/.next/server/app/index.segments/__PAGE__.segment.rsc +0 -5
  417. package/dist/.next/server/app/index.segments/_full.segment.rsc +0 -15
  418. package/dist/.next/server/app/index.segments/_head.segment.rsc +0 -6
  419. package/dist/.next/server/app/index.segments/_index.segment.rsc +0 -5
  420. package/dist/.next/server/app/index.segments/_tree.segment.rsc +0 -3
  421. package/dist/.next/server/app/page/app-paths-manifest.json +0 -3
  422. package/dist/.next/server/app/page/build-manifest.json +0 -16
  423. package/dist/.next/server/app/page/next-font-manifest.json +0 -10
  424. package/dist/.next/server/app/page/react-loadable-manifest.json +0 -1
  425. package/dist/.next/server/app/page/server-reference-manifest.json +0 -4
  426. package/dist/.next/server/app/page.js +0 -14
  427. package/dist/.next/server/app/page.js.map +0 -5
  428. package/dist/.next/server/app/page.js.nft.json +0 -1
  429. package/dist/.next/server/app/page_client-reference-manifest.js +0 -3
  430. package/dist/.next/server/app-paths-manifest.json +0 -6
  431. package/dist/.next/server/chunks/[externals]_next_dist_0arv.vj._.js +0 -3
  432. package/dist/.next/server/chunks/[root-of-the-server]__0vcj1q1._.js +0 -13
  433. package/dist/.next/server/chunks/[turbopack]_runtime.js +0 -903
  434. package/dist/.next/server/chunks/_next-internal_server_app_icon_svg_route_actions_0-0ehc~.js +0 -3
  435. package/dist/.next/server/chunks/ssr/05w9_next_dist_0ihu0u9._.js +0 -6
  436. package/dist/.next/server/chunks/ssr/05w9_next_dist_client_components_12u3mib._.js +0 -3
  437. package/dist/.next/server/chunks/ssr/05w9_next_dist_client_components_builtin_forbidden_04fbe_..js +0 -3
  438. package/dist/.next/server/chunks/ssr/05w9_next_dist_client_components_builtin_global-error_0brpl_..js +0 -3
  439. package/dist/.next/server/chunks/ssr/05w9_next_dist_client_components_builtin_unauthorized_0~2g66g.js +0 -3
  440. package/dist/.next/server/chunks/ssr/05w9_next_dist_esm_build_templates_app-page_0~cyr1_.js +0 -4
  441. package/dist/.next/server/chunks/ssr/05w9_next_dist_esm_build_templates_app-page_1105emf.js +0 -4
  442. package/dist/.next/server/chunks/ssr/05w9_next_dist_esm_build_templates_app-page_11uhyqv.js +0 -4
  443. package/dist/.next/server/chunks/ssr/[root-of-the-server]__0.t9_75._.js +0 -33
  444. package/dist/.next/server/chunks/ssr/[root-of-the-server]__0c0ud_z._.js +0 -3
  445. package/dist/.next/server/chunks/ssr/[root-of-the-server]__0f9_8d4._.js +0 -3
  446. package/dist/.next/server/chunks/ssr/[root-of-the-server]__0l5ko41._.js +0 -19
  447. package/dist/.next/server/chunks/ssr/[root-of-the-server]__0mn6z7i._.js +0 -3
  448. package/dist/.next/server/chunks/ssr/[root-of-the-server]__0npxxst._.js +0 -33
  449. package/dist/.next/server/chunks/ssr/[root-of-the-server]__0qjhaca._.js +0 -3
  450. package/dist/.next/server/chunks/ssr/[root-of-the-server]__0rwgw3s._.js +0 -3
  451. package/dist/.next/server/chunks/ssr/[turbopack]_runtime.js +0 -903
  452. package/dist/.next/server/chunks/ssr/_next-internal_server_app__global-error_page_actions_0k77kol.js +0 -3
  453. package/dist/.next/server/chunks/ssr/_next-internal_server_app__not-found_page_actions_0eq97pa.js +0 -3
  454. package/dist/.next/server/chunks/ssr/_next-internal_server_app_page_actions_09-gtaw.js +0 -3
  455. package/dist/.next/server/chunks/ssr/node_modules__pnpm_056~6.6._.js +0 -3
  456. package/dist/.next/server/chunks/ssr/node_modules__pnpm_0~j0k.e._.js +0 -33
  457. package/dist/.next/server/functions-config-manifest.json +0 -4
  458. package/dist/.next/server/middleware-build-manifest.js +0 -20
  459. package/dist/.next/server/middleware-manifest.json +0 -6
  460. package/dist/.next/server/next-font-manifest.js +0 -1
  461. package/dist/.next/server/next-font-manifest.json +0 -13
  462. package/dist/.next/server/pages/404.html +0 -1
  463. package/dist/.next/server/pages/500.html +0 -1
  464. package/dist/.next/server/pages-manifest.json +0 -4
  465. package/dist/.next/server/prefetch-hints.json +0 -1
  466. package/dist/.next/server/server-reference-manifest.js +0 -1
  467. package/dist/.next/server/server-reference-manifest.json +0 -5
  468. package/dist/.next/static/SMWpxFVvkpYFxY7uuFvGB/_buildManifest.js +0 -11
  469. package/dist/.next/static/SMWpxFVvkpYFxY7uuFvGB/_clientMiddlewareManifest.js +0 -1
  470. package/dist/.next/static/SMWpxFVvkpYFxY7uuFvGB/_ssgManifest.js +0 -1
  471. package/dist/.next/static/chunks/01qk2~bgf76vu.js +0 -1
  472. package/dist/.next/static/chunks/03~yq9q893hmn.js +0 -1
  473. package/dist/.next/static/chunks/080queev.r2uy.js +0 -31
  474. package/dist/.next/static/chunks/0v3lyuj75aq50.js +0 -1
  475. package/dist/.next/static/chunks/10b~xdx5c-i7s.js +0 -5
  476. package/dist/.next/static/chunks/14gla2ascffgv.css +0 -2
  477. package/dist/.next/static/chunks/turbopack-0m-970~qvs7sc.js +0 -1
  478. package/dist/.next/static/media/7178b3e590c64307-s.11.cyxs5p-0z~.woff2 +0 -0
  479. package/dist/.next/static/media/8a480f0b521d4e75-s.06d3mdzz5bre_.woff2 +0 -0
  480. package/dist/.next/static/media/caa3a2e1cccd8315-s.p.16t1db8_9y2o~.woff2 +0 -0
  481. package/dist/package.json +0 -88
  482. package/dist/server.js +0 -38
  483. /package/{dist/.next/server/app/icon.svg.body → backend/src/flowent/static/favicon.svg} +0 -0
  484. /package/dist/{.next/static/media/icon.0.r~afrtrocz9.svg → frontend/favicon.svg} +0 -0
@@ -0,0 +1,356 @@
1
+ from __future__ import annotations
2
+
3
+ import base64
4
+ import json
5
+ import struct
6
+ import tempfile
7
+ import uuid
8
+ from dataclasses import dataclass
9
+ from pathlib import Path
10
+
11
+ from flowent.state_db import (
12
+ get_images_dir,
13
+ get_legacy_image_assets_dir,
14
+ open_state_db,
15
+ )
16
+
17
+ ALLOWED_IMAGE_MIME_TYPES = frozenset(
18
+ {
19
+ "image/png",
20
+ "image/jpeg",
21
+ "image/gif",
22
+ "image/webp",
23
+ }
24
+ )
25
+
26
+ _IMAGE_EXTENSIONS = {
27
+ "image/png": ".png",
28
+ "image/jpeg": ".jpg",
29
+ "image/gif": ".gif",
30
+ "image/webp": ".webp",
31
+ }
32
+
33
+
34
+ @dataclass(frozen=True)
35
+ class ImageAsset:
36
+ id: str
37
+ stored_name: str
38
+ mime_type: str
39
+ width: int | None = None
40
+ height: int | None = None
41
+ original_name: str | None = None
42
+
43
+ @property
44
+ def file_path(self) -> Path:
45
+ return _get_assets_dir() / self.stored_name
46
+
47
+ @property
48
+ def url_path(self) -> str:
49
+ return f"/api/image-assets/{self.id}"
50
+
51
+ def serialize(self) -> dict[str, object]:
52
+ return {
53
+ "id": self.id,
54
+ "stored_name": self.stored_name,
55
+ "mime_type": self.mime_type,
56
+ "width": self.width,
57
+ "height": self.height,
58
+ "original_name": self.original_name,
59
+ "url": self.url_path,
60
+ }
61
+
62
+
63
+ def _get_assets_dir() -> Path:
64
+ return get_images_dir()
65
+
66
+
67
+ def _get_legacy_metadata_path(asset_id: str) -> Path:
68
+ return get_legacy_image_assets_dir() / f"{asset_id}.json"
69
+
70
+
71
+ def _get_legacy_file_path(stored_name: str) -> Path:
72
+ return get_legacy_image_assets_dir() / stored_name
73
+
74
+
75
+ def _detect_mime_type(data: bytes) -> str | None:
76
+ if data.startswith(b"\x89PNG\r\n\x1a\n"):
77
+ return "image/png"
78
+ if data.startswith(b"\xff\xd8\xff"):
79
+ return "image/jpeg"
80
+ if data.startswith((b"GIF87a", b"GIF89a")):
81
+ return "image/gif"
82
+ if data.startswith(b"RIFF") and data[8:12] == b"WEBP":
83
+ return "image/webp"
84
+ return None
85
+
86
+
87
+ def _normalize_mime_type(data: bytes, mime_type: str | None) -> str:
88
+ normalized = mime_type.strip().lower() if isinstance(mime_type, str) else ""
89
+ if normalized in ALLOWED_IMAGE_MIME_TYPES:
90
+ return normalized
91
+ detected = _detect_mime_type(data)
92
+ if detected is not None:
93
+ return detected
94
+ raise ValueError("Unsupported image type. Allowed types: PNG, JPEG, GIF, WEBP.")
95
+
96
+
97
+ def _read_png_size(data: bytes) -> tuple[int | None, int | None]:
98
+ if len(data) < 24:
99
+ return None, None
100
+ if data[12:16] != b"IHDR":
101
+ return None, None
102
+ width = struct.unpack(">I", data[16:20])[0]
103
+ height = struct.unpack(">I", data[20:24])[0]
104
+ return width, height
105
+
106
+
107
+ def _read_gif_size(data: bytes) -> tuple[int | None, int | None]:
108
+ if len(data) < 10:
109
+ return None, None
110
+ width, height = struct.unpack("<HH", data[6:10])
111
+ return width, height
112
+
113
+
114
+ def _read_jpeg_size(data: bytes) -> tuple[int | None, int | None]:
115
+ if len(data) < 4 or not data.startswith(b"\xff\xd8"):
116
+ return None, None
117
+ offset = 2
118
+ while offset + 1 < len(data):
119
+ if data[offset] != 0xFF:
120
+ offset += 1
121
+ continue
122
+ marker = data[offset + 1]
123
+ offset += 2
124
+ if marker in {0xD8, 0xD9}:
125
+ continue
126
+ if offset + 2 > len(data):
127
+ break
128
+ segment_length = struct.unpack(">H", data[offset : offset + 2])[0]
129
+ if segment_length < 2 or offset + segment_length > len(data):
130
+ break
131
+ if marker in {
132
+ 0xC0,
133
+ 0xC1,
134
+ 0xC2,
135
+ 0xC3,
136
+ 0xC5,
137
+ 0xC6,
138
+ 0xC7,
139
+ 0xC9,
140
+ 0xCA,
141
+ 0xCB,
142
+ 0xCD,
143
+ 0xCE,
144
+ 0xCF,
145
+ }:
146
+ if offset + 7 > len(data):
147
+ break
148
+ height = struct.unpack(">H", data[offset + 3 : offset + 5])[0]
149
+ width = struct.unpack(">H", data[offset + 5 : offset + 7])[0]
150
+ return width, height
151
+ offset += segment_length
152
+ return None, None
153
+
154
+
155
+ def _read_webp_size(data: bytes) -> tuple[int | None, int | None]:
156
+ if len(data) < 30 or not data.startswith(b"RIFF") or data[8:12] != b"WEBP":
157
+ return None, None
158
+ chunk_type = data[12:16]
159
+ if chunk_type == b"VP8X" and len(data) >= 30:
160
+ width = 1 + int.from_bytes(data[24:27], "little")
161
+ height = 1 + int.from_bytes(data[27:30], "little")
162
+ return width, height
163
+ if chunk_type == b"VP8L" and len(data) >= 25:
164
+ value = int.from_bytes(data[21:25], "little")
165
+ width = (value & 0x3FFF) + 1
166
+ height = ((value >> 14) & 0x3FFF) + 1
167
+ return width, height
168
+ return None, None
169
+
170
+
171
+ def _read_image_size(data: bytes, mime_type: str) -> tuple[int | None, int | None]:
172
+ if mime_type == "image/png":
173
+ return _read_png_size(data)
174
+ if mime_type == "image/jpeg":
175
+ return _read_jpeg_size(data)
176
+ if mime_type == "image/gif":
177
+ return _read_gif_size(data)
178
+ if mime_type == "image/webp":
179
+ return _read_webp_size(data)
180
+ return None, None
181
+
182
+
183
+ def create_image_asset(
184
+ data: bytes,
185
+ *,
186
+ mime_type: str | None = None,
187
+ original_name: str | None = None,
188
+ ) -> ImageAsset:
189
+ if not data:
190
+ raise ValueError("Image file is empty.")
191
+ normalized_mime_type = _normalize_mime_type(data, mime_type)
192
+ width, height = _read_image_size(data, normalized_mime_type)
193
+ asset_id = str(uuid.uuid4())
194
+ extension = _IMAGE_EXTENSIONS[normalized_mime_type]
195
+ stored_name = f"{asset_id}{extension}"
196
+ asset = ImageAsset(
197
+ id=asset_id,
198
+ stored_name=stored_name,
199
+ mime_type=normalized_mime_type,
200
+ width=width,
201
+ height=height,
202
+ original_name=original_name or None,
203
+ )
204
+ asset_dir = _get_assets_dir()
205
+ asset_dir.mkdir(parents=True, exist_ok=True)
206
+
207
+ with tempfile.NamedTemporaryFile("wb", dir=asset_dir, delete=False) as handle:
208
+ handle.write(data)
209
+ temp_file_path = Path(handle.name)
210
+ temp_file_path.replace(asset.file_path)
211
+ _persist_image_asset(asset)
212
+ return asset
213
+
214
+
215
+ def get_image_asset(asset_id: str) -> ImageAsset | None:
216
+ if not asset_id.strip():
217
+ return None
218
+ asset = _load_persisted_image_asset(asset_id)
219
+ if asset is not None:
220
+ if asset.file_path.is_file():
221
+ return asset
222
+ if _restore_legacy_image_file(asset):
223
+ return asset
224
+ return None
225
+ legacy_asset = _load_legacy_image_asset(asset_id)
226
+ if legacy_asset is None:
227
+ return None
228
+ _persist_image_asset(legacy_asset)
229
+ if _restore_legacy_image_file(legacy_asset):
230
+ return legacy_asset
231
+ return None
232
+
233
+
234
+ def _persist_image_asset(asset: ImageAsset) -> None:
235
+ connection = open_state_db(create=True)
236
+ assert connection is not None
237
+ try:
238
+ with connection:
239
+ connection.execute(
240
+ """
241
+ INSERT OR REPLACE INTO image_assets (
242
+ id,
243
+ stored_name,
244
+ mime_type,
245
+ width,
246
+ height,
247
+ original_name
248
+ ) VALUES (?, ?, ?, ?, ?, ?)
249
+ """,
250
+ (
251
+ asset.id,
252
+ asset.stored_name,
253
+ asset.mime_type,
254
+ asset.width,
255
+ asset.height,
256
+ asset.original_name,
257
+ ),
258
+ )
259
+ finally:
260
+ connection.close()
261
+
262
+
263
+ def _load_persisted_image_asset(asset_id: str) -> ImageAsset | None:
264
+ connection = open_state_db(create=False)
265
+ if connection is None:
266
+ return None
267
+ try:
268
+ row = connection.execute(
269
+ """
270
+ SELECT id, stored_name, mime_type, width, height, original_name
271
+ FROM image_assets
272
+ WHERE id = ?
273
+ """,
274
+ (asset_id,),
275
+ ).fetchone()
276
+ finally:
277
+ connection.close()
278
+ if row is None:
279
+ return None
280
+ return ImageAsset(
281
+ id=row["id"],
282
+ stored_name=row["stored_name"],
283
+ mime_type=row["mime_type"],
284
+ width=row["width"] if isinstance(row["width"], int) else None,
285
+ height=row["height"] if isinstance(row["height"], int) else None,
286
+ original_name=(
287
+ row["original_name"] if isinstance(row["original_name"], str) else None
288
+ ),
289
+ )
290
+
291
+
292
+ def _load_legacy_image_asset(asset_id: str) -> ImageAsset | None:
293
+ metadata_path = _get_legacy_metadata_path(asset_id)
294
+ if not metadata_path.is_file():
295
+ return None
296
+ raw = json.loads(metadata_path.read_text(encoding="utf-8"))
297
+ if not isinstance(raw, dict):
298
+ return None
299
+ stored_name = raw.get("stored_name")
300
+ mime_type = raw.get("mime_type")
301
+ if not isinstance(stored_name, str) or not isinstance(mime_type, str):
302
+ return None
303
+ if not _get_legacy_file_path(stored_name).is_file():
304
+ return None
305
+ return ImageAsset(
306
+ id=asset_id,
307
+ stored_name=stored_name,
308
+ mime_type=mime_type,
309
+ width=raw.get("width") if isinstance(raw.get("width"), int) else None,
310
+ height=raw.get("height") if isinstance(raw.get("height"), int) else None,
311
+ original_name=(
312
+ raw.get("original_name")
313
+ if isinstance(raw.get("original_name"), str)
314
+ else None
315
+ ),
316
+ )
317
+
318
+
319
+ def _restore_legacy_image_file(asset: ImageAsset) -> bool:
320
+ target_path = asset.file_path
321
+ if target_path.is_file():
322
+ return True
323
+ legacy_file_path = _get_legacy_file_path(asset.stored_name)
324
+ if not legacy_file_path.is_file():
325
+ return False
326
+ asset_dir = _get_assets_dir()
327
+ asset_dir.mkdir(parents=True, exist_ok=True)
328
+ with tempfile.NamedTemporaryFile("wb", dir=asset_dir, delete=False) as handle:
329
+ handle.write(legacy_file_path.read_bytes())
330
+ temp_file_path = Path(handle.name)
331
+ temp_file_path.replace(target_path)
332
+ return True
333
+
334
+
335
+ def require_image_asset(asset_id: str) -> ImageAsset:
336
+ asset = get_image_asset(asset_id)
337
+ if asset is None:
338
+ raise ValueError(f"Unknown image asset `{asset_id}`.")
339
+ return asset
340
+
341
+
342
+ def read_image_asset_bytes(asset_id: str) -> tuple[ImageAsset, bytes]:
343
+ asset = require_image_asset(asset_id)
344
+ return asset, asset.file_path.read_bytes()
345
+
346
+
347
+ def encode_image_asset_as_data_url(asset_id: str) -> tuple[ImageAsset, str]:
348
+ asset, data = read_image_asset_bytes(asset_id)
349
+ encoded = base64.b64encode(data).decode("ascii")
350
+ return asset, f"data:{asset.mime_type};base64,{encoded}"
351
+
352
+
353
+ def encode_image_asset_as_base64(asset_id: str) -> tuple[ImageAsset, str]:
354
+ asset, data = read_image_asset_bytes(asset_id)
355
+ encoded = base64.b64encode(data).decode("ascii")
356
+ return asset, encoded
@@ -0,0 +1,155 @@
1
+ from __future__ import annotations
2
+
3
+ import logging
4
+ import sys
5
+ import time
6
+ from collections.abc import Sequence
7
+ from datetime import datetime
8
+ from pathlib import Path
9
+ from typing import TYPE_CHECKING, Literal
10
+
11
+ from loguru import logger
12
+
13
+ if TYPE_CHECKING:
14
+ from loguru import Record
15
+
16
+ from flowent.config import Config
17
+
18
+ RuntimeMode = Literal["dev", "release"]
19
+
20
+ CONSOLE_FORMAT = (
21
+ "<green>{time:HH:mm:ss.SSS}</green> | "
22
+ "<level>{level: <8}</level> | "
23
+ "<cyan>{extra[source]}</cyan>{extra[agent_suffix]} | "
24
+ "<level>{message}</level>{exception}"
25
+ )
26
+
27
+ FILE_FORMAT = (
28
+ "{time:YYYY-MM-DD HH:mm:ss.SSS} | "
29
+ "{level: <8} | "
30
+ "{extra[source]}{extra[agent_suffix]} | "
31
+ "{message}{exception}"
32
+ )
33
+
34
+
35
+ class HealthcheckAccessFilter(logging.Filter):
36
+ def filter(self, record: logging.LogRecord) -> bool:
37
+ args = record.args
38
+ if not isinstance(args, tuple) or len(args) < 3:
39
+ return True
40
+ full_path = args[2]
41
+ if not isinstance(full_path, str):
42
+ return True
43
+ return full_path.split("?", 1)[0] != "/health"
44
+
45
+
46
+ def detect_runtime_mode(
47
+ argv: Sequence[str] | None = None,
48
+ config: Config | None = None,
49
+ ) -> RuntimeMode:
50
+ args = list(sys.argv if argv is None else argv)
51
+ executable = Path(args[0]).name.lower() if args else ""
52
+ command = args[1] if len(args) > 1 else ""
53
+
54
+ if "fastapi" in executable and command == "dev":
55
+ return "dev"
56
+ if "fastapi" in executable and command == "run":
57
+ return "release"
58
+ if command == "dev":
59
+ return "dev"
60
+ if command == "run":
61
+ return "release"
62
+ if "--reload" in args:
63
+ return "dev"
64
+ if config is not None and config.DEBUG:
65
+ return "dev"
66
+ return "release"
67
+
68
+
69
+ def _patch_record(record: Record) -> None:
70
+ agent_id = record["extra"].get("agent_id")
71
+ record["extra"]["agent_suffix"] = f" | agent:{agent_id}" if agent_id else ""
72
+ record["extra"]["source"] = (
73
+ f"{record['name']}:{record['function']}:{record['line']}"
74
+ )
75
+
76
+
77
+ def _get_log_dir(runtime_dir: Path | None = None) -> Path:
78
+ base_dir = Path.cwd() if runtime_dir is None else runtime_dir
79
+ return base_dir / "logs"
80
+
81
+
82
+ def _allocate_log_file(log_dir: Path) -> Path:
83
+ timestamp = time.time_ns() // 1_000_000
84
+ prefix = datetime.fromtimestamp(timestamp / 1000).strftime("%Y-%m-%d_%H%M%S")
85
+ log_path = log_dir / f"{prefix}_{timestamp}.log"
86
+ while log_path.exists():
87
+ timestamp += 1
88
+ prefix = datetime.fromtimestamp(timestamp / 1000).strftime("%Y-%m-%d_%H%M%S")
89
+ log_path = log_dir / f"{prefix}_{timestamp}.log"
90
+ return log_path
91
+
92
+
93
+ def _sort_key(path: Path) -> tuple[int, int, str]:
94
+ try:
95
+ timestamp = int(path.stem.rsplit("_", 1)[-1])
96
+ except ValueError:
97
+ try:
98
+ timestamp = int(path.stem)
99
+ except ValueError:
100
+ timestamp = -1
101
+ return (timestamp, path.stat().st_mtime_ns, path.name)
102
+
103
+
104
+ def prune_old_logs(log_dir: Path, keep: int = 10) -> None:
105
+ log_files = [path for path in log_dir.iterdir() if path.is_file()]
106
+ log_files.sort(key=_sort_key, reverse=True)
107
+ for path in log_files[keep:]:
108
+ path.unlink(missing_ok=True)
109
+
110
+
111
+ def configure_uvicorn_access_logging() -> None:
112
+ access_logger = logging.getLogger("uvicorn.access")
113
+ if any(isinstance(f, HealthcheckAccessFilter) for f in access_logger.filters):
114
+ return
115
+ access_logger.addFilter(HealthcheckAccessFilter())
116
+
117
+
118
+ def setup_logging(
119
+ config: Config | None = None,
120
+ *,
121
+ argv: Sequence[str] | None = None,
122
+ runtime_dir: Path | None = None,
123
+ ) -> None:
124
+ if config is None:
125
+ config = Config()
126
+
127
+ mode = detect_runtime_mode(argv=argv, config=config)
128
+ console_level = "DEBUG" if mode == "dev" else "INFO"
129
+
130
+ logger.remove()
131
+ logger.configure(patcher=_patch_record)
132
+ configure_uvicorn_access_logging()
133
+
134
+ log_dir = _get_log_dir(runtime_dir)
135
+ log_dir.mkdir(parents=True, exist_ok=True)
136
+ log_path = _allocate_log_file(log_dir)
137
+ log_path.touch()
138
+ prune_old_logs(log_dir)
139
+
140
+ logger.add(
141
+ sys.stderr,
142
+ format=CONSOLE_FORMAT,
143
+ level=console_level,
144
+ colorize=True,
145
+ backtrace=True,
146
+ diagnose=mode == "dev",
147
+ )
148
+
149
+ logger.add(
150
+ log_path,
151
+ format=FILE_FORMAT,
152
+ level="TRACE",
153
+ backtrace=True,
154
+ diagnose=mode == "dev",
155
+ )
@@ -0,0 +1,124 @@
1
+ from __future__ import annotations
2
+
3
+ import asyncio
4
+ import os
5
+ from contextlib import asynccontextmanager
6
+ from pathlib import Path
7
+
8
+ from fastapi import FastAPI
9
+ from fastapi.responses import FileResponse
10
+ from fastapi.staticfiles import StaticFiles
11
+ from starlette.middleware.sessions import SessionMiddleware
12
+
13
+ from flowent.access import AccessControlMiddleware
14
+ from flowent.config import Config
15
+ from flowent.events import event_bus
16
+ from flowent.logging import setup_logging
17
+ from flowent.runtime import bootstrap_runtime, shutdown_runtime
18
+
19
+ config = Config()
20
+ setup_logging(config)
21
+
22
+ DEFAULT_STATIC_DIR = Path(__file__).parent / "static"
23
+
24
+
25
+ def frontend_static_directory() -> Path:
26
+ configured_directory = os.environ.get("FLOWENT_STATIC_DIR")
27
+ if configured_directory:
28
+ return Path(configured_directory)
29
+ repository_frontend_dist = Path(__file__).resolve().parents[3] / "frontend" / "dist"
30
+ if repository_frontend_dist.is_dir():
31
+ return repository_frontend_dist
32
+ return DEFAULT_STATIC_DIR
33
+
34
+
35
+ @asynccontextmanager
36
+ async def lifespan(app: FastAPI):
37
+ from flowent.access import (
38
+ initialize_live_access_signature,
39
+ refresh_live_access_signature,
40
+ )
41
+
42
+ loop = asyncio.get_running_loop()
43
+ event_bus.set_loop(loop)
44
+
45
+ bootstrap_runtime()
46
+ initialize_live_access_signature()
47
+
48
+ stop_event = asyncio.Event()
49
+
50
+ async def watch_access_state() -> None:
51
+ while True:
52
+ try:
53
+ await asyncio.wait_for(stop_event.wait(), timeout=1.0)
54
+ return
55
+ except TimeoutError:
56
+ if refresh_live_access_signature():
57
+ await event_bus.close_all(
58
+ code=4001, reason="Access session updated"
59
+ )
60
+
61
+ access_watch_task = asyncio.create_task(watch_access_state())
62
+
63
+ yield
64
+
65
+ stop_event.set()
66
+ await access_watch_task
67
+ shutdown_runtime()
68
+
69
+
70
+ def create_app(*, serve_frontend: bool = True) -> FastAPI:
71
+ from flowent.access import ensure_session_signing_secret
72
+ from flowent.settings import get_settings, save_settings
73
+
74
+ settings = get_settings()
75
+ if ensure_session_signing_secret(settings):
76
+ save_settings(settings)
77
+ session_secret = (
78
+ config.SESSION_SECRET.strip() or settings.access.session_signing_secret
79
+ )
80
+
81
+ app = FastAPI(
82
+ title=config.APP_NAME,
83
+ debug=config.DEBUG,
84
+ lifespan=lifespan,
85
+ )
86
+ app.add_middleware(AccessControlMiddleware)
87
+ app.add_middleware(
88
+ SessionMiddleware,
89
+ secret_key=session_secret,
90
+ session_cookie=config.SESSION_COOKIE_NAME,
91
+ max_age=config.SESSION_MAX_AGE_SECONDS,
92
+ same_site="lax",
93
+ https_only=False,
94
+ )
95
+
96
+ from flowent.routes import router
97
+
98
+ app.include_router(router)
99
+ static_dir = frontend_static_directory().resolve(strict=False)
100
+
101
+ if serve_frontend and static_dir.is_dir():
102
+ app.mount(
103
+ "/assets",
104
+ StaticFiles(directory=static_dir / "assets"),
105
+ name="assets",
106
+ )
107
+
108
+ @app.get("/{path:path}")
109
+ async def spa_fallback(path: str) -> FileResponse:
110
+ file = (static_dir / path).resolve(strict=False)
111
+ if file.is_file() and file.is_relative_to(static_dir):
112
+ return FileResponse(file)
113
+ return FileResponse(static_dir / "index.html")
114
+
115
+ return app
116
+
117
+
118
+ app = create_app()
119
+
120
+
121
+ if __name__ == "__main__":
122
+ import uvicorn
123
+
124
+ uvicorn.run(app, reload=config.DEBUG)