machinaos 0.0.1

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 (288) hide show
  1. package/.env.template +71 -0
  2. package/LICENSE +21 -0
  3. package/README.md +87 -0
  4. package/bin/cli.js +159 -0
  5. package/client/.dockerignore +45 -0
  6. package/client/Dockerfile +68 -0
  7. package/client/eslint.config.js +29 -0
  8. package/client/index.html +13 -0
  9. package/client/nginx.conf +66 -0
  10. package/client/package.json +48 -0
  11. package/client/src/App.tsx +27 -0
  12. package/client/src/Dashboard.tsx +1173 -0
  13. package/client/src/ParameterPanel.tsx +301 -0
  14. package/client/src/components/AIAgentNode.tsx +321 -0
  15. package/client/src/components/APIKeyValidator.tsx +118 -0
  16. package/client/src/components/ClaudeChatModelNode.tsx +18 -0
  17. package/client/src/components/ConditionalEdge.tsx +189 -0
  18. package/client/src/components/CredentialsModal.tsx +306 -0
  19. package/client/src/components/EdgeConditionEditor.tsx +443 -0
  20. package/client/src/components/GeminiChatModelNode.tsx +18 -0
  21. package/client/src/components/GenericNode.tsx +357 -0
  22. package/client/src/components/LocationParameterPanel.tsx +154 -0
  23. package/client/src/components/ModelNode.tsx +286 -0
  24. package/client/src/components/OpenAIChatModelNode.tsx +18 -0
  25. package/client/src/components/OutputPanel.tsx +471 -0
  26. package/client/src/components/ParameterRenderer.tsx +1874 -0
  27. package/client/src/components/SkillEditorModal.tsx +417 -0
  28. package/client/src/components/SquareNode.tsx +797 -0
  29. package/client/src/components/StartNode.tsx +250 -0
  30. package/client/src/components/ToolkitNode.tsx +365 -0
  31. package/client/src/components/TriggerNode.tsx +463 -0
  32. package/client/src/components/auth/LoginPage.tsx +247 -0
  33. package/client/src/components/auth/ProtectedRoute.tsx +59 -0
  34. package/client/src/components/base/BaseChatModelNode.tsx +271 -0
  35. package/client/src/components/icons/AIProviderIcons.tsx +50 -0
  36. package/client/src/components/maps/GoogleMapsPicker.tsx +137 -0
  37. package/client/src/components/maps/MapsPreviewPanel.tsx +110 -0
  38. package/client/src/components/maps/index.ts +26 -0
  39. package/client/src/components/parameterPanel/InputSection.tsx +1094 -0
  40. package/client/src/components/parameterPanel/LocationPanelLayout.tsx +65 -0
  41. package/client/src/components/parameterPanel/MapsSection.tsx +92 -0
  42. package/client/src/components/parameterPanel/MiddleSection.tsx +571 -0
  43. package/client/src/components/parameterPanel/OutputSection.tsx +81 -0
  44. package/client/src/components/parameterPanel/ParameterPanelLayout.tsx +82 -0
  45. package/client/src/components/parameterPanel/ToolSchemaEditor.tsx +436 -0
  46. package/client/src/components/parameterPanel/index.ts +42 -0
  47. package/client/src/components/shared/DataPanel.tsx +142 -0
  48. package/client/src/components/shared/JSONTreeRenderer.tsx +106 -0
  49. package/client/src/components/ui/AIResultModal.tsx +204 -0
  50. package/client/src/components/ui/AndroidSettingsPanel.tsx +401 -0
  51. package/client/src/components/ui/CodeEditor.tsx +81 -0
  52. package/client/src/components/ui/CollapsibleSection.tsx +88 -0
  53. package/client/src/components/ui/ComponentItem.tsx +154 -0
  54. package/client/src/components/ui/ComponentPalette.tsx +321 -0
  55. package/client/src/components/ui/ConsolePanel.tsx +1074 -0
  56. package/client/src/components/ui/ErrorBoundary.tsx +196 -0
  57. package/client/src/components/ui/InputNodesPanel.tsx +204 -0
  58. package/client/src/components/ui/MapSelector.tsx +314 -0
  59. package/client/src/components/ui/Modal.tsx +149 -0
  60. package/client/src/components/ui/NodeContextMenu.tsx +192 -0
  61. package/client/src/components/ui/NodeOutputPanel.tsx +1150 -0
  62. package/client/src/components/ui/OutputDisplayPanel.tsx +381 -0
  63. package/client/src/components/ui/SettingsPanel.tsx +243 -0
  64. package/client/src/components/ui/TopToolbar.tsx +736 -0
  65. package/client/src/components/ui/WhatsAppSettingsPanel.tsx +345 -0
  66. package/client/src/components/ui/WorkflowSidebar.tsx +294 -0
  67. package/client/src/config/antdTheme.ts +186 -0
  68. package/client/src/config/api.ts +54 -0
  69. package/client/src/contexts/AuthContext.tsx +221 -0
  70. package/client/src/contexts/ThemeContext.tsx +42 -0
  71. package/client/src/contexts/WebSocketContext.tsx +1971 -0
  72. package/client/src/factories/baseChatModelFactory.ts +256 -0
  73. package/client/src/hooks/useAndroidOperations.ts +164 -0
  74. package/client/src/hooks/useApiKeyValidation.ts +107 -0
  75. package/client/src/hooks/useApiKeys.ts +238 -0
  76. package/client/src/hooks/useAppTheme.ts +17 -0
  77. package/client/src/hooks/useComponentPalette.ts +51 -0
  78. package/client/src/hooks/useCopyPaste.ts +155 -0
  79. package/client/src/hooks/useDragAndDrop.ts +124 -0
  80. package/client/src/hooks/useDragVariable.ts +88 -0
  81. package/client/src/hooks/useExecution.ts +313 -0
  82. package/client/src/hooks/useParameterPanel.ts +176 -0
  83. package/client/src/hooks/useReactFlowNodes.ts +189 -0
  84. package/client/src/hooks/useToolSchema.ts +209 -0
  85. package/client/src/hooks/useWhatsApp.ts +196 -0
  86. package/client/src/hooks/useWorkflowManagement.ts +46 -0
  87. package/client/src/index.css +315 -0
  88. package/client/src/main.tsx +19 -0
  89. package/client/src/nodeDefinitions/aiAgentNodes.ts +336 -0
  90. package/client/src/nodeDefinitions/aiModelNodes.ts +340 -0
  91. package/client/src/nodeDefinitions/androidDeviceNodes.ts +140 -0
  92. package/client/src/nodeDefinitions/androidServiceNodes.ts +383 -0
  93. package/client/src/nodeDefinitions/chatNodes.ts +135 -0
  94. package/client/src/nodeDefinitions/codeNodes.ts +54 -0
  95. package/client/src/nodeDefinitions/documentNodes.ts +379 -0
  96. package/client/src/nodeDefinitions/index.ts +15 -0
  97. package/client/src/nodeDefinitions/locationNodes.ts +463 -0
  98. package/client/src/nodeDefinitions/schedulerNodes.ts +220 -0
  99. package/client/src/nodeDefinitions/skillNodes.ts +211 -0
  100. package/client/src/nodeDefinitions/toolNodes.ts +198 -0
  101. package/client/src/nodeDefinitions/utilityNodes.ts +284 -0
  102. package/client/src/nodeDefinitions/whatsappNodes.ts +865 -0
  103. package/client/src/nodeDefinitions/workflowNodes.ts +41 -0
  104. package/client/src/nodeDefinitions.ts +104 -0
  105. package/client/src/schemas/workflowSchema.ts +264 -0
  106. package/client/src/services/dynamicParameterService.ts +96 -0
  107. package/client/src/services/execution/aiAgentExecutionService.ts +35 -0
  108. package/client/src/services/executionService.ts +232 -0
  109. package/client/src/services/workflowApi.ts +91 -0
  110. package/client/src/store/useAppStore.ts +582 -0
  111. package/client/src/styles/theme.ts +508 -0
  112. package/client/src/styles/zIndex.ts +17 -0
  113. package/client/src/types/ComponentTypes.ts +39 -0
  114. package/client/src/types/EdgeCondition.ts +231 -0
  115. package/client/src/types/INodeProperties.ts +288 -0
  116. package/client/src/types/NodeTypes.ts +28 -0
  117. package/client/src/utils/formatters.ts +33 -0
  118. package/client/src/utils/googleMapsLoader.ts +140 -0
  119. package/client/src/utils/locationUtils.ts +85 -0
  120. package/client/src/utils/nodeUtils.ts +31 -0
  121. package/client/src/utils/workflow.ts +30 -0
  122. package/client/src/utils/workflowExport.ts +120 -0
  123. package/client/src/vite-env.d.ts +12 -0
  124. package/client/tailwind.config.js +60 -0
  125. package/client/tsconfig.json +25 -0
  126. package/client/tsconfig.node.json +11 -0
  127. package/client/vite.config.js +35 -0
  128. package/docker-compose.prod.yml +107 -0
  129. package/docker-compose.yml +104 -0
  130. package/docs-MachinaOs/README.md +85 -0
  131. package/docs-MachinaOs/deployment/docker.mdx +228 -0
  132. package/docs-MachinaOs/deployment/production.mdx +345 -0
  133. package/docs-MachinaOs/docs.json +75 -0
  134. package/docs-MachinaOs/faq.mdx +309 -0
  135. package/docs-MachinaOs/favicon.svg +5 -0
  136. package/docs-MachinaOs/installation.mdx +160 -0
  137. package/docs-MachinaOs/introduction.mdx +114 -0
  138. package/docs-MachinaOs/logo/dark.svg +6 -0
  139. package/docs-MachinaOs/logo/light.svg +6 -0
  140. package/docs-MachinaOs/nodes/ai-agent.mdx +216 -0
  141. package/docs-MachinaOs/nodes/ai-models.mdx +240 -0
  142. package/docs-MachinaOs/nodes/android.mdx +411 -0
  143. package/docs-MachinaOs/nodes/overview.mdx +181 -0
  144. package/docs-MachinaOs/nodes/schedulers.mdx +316 -0
  145. package/docs-MachinaOs/nodes/webhooks.mdx +330 -0
  146. package/docs-MachinaOs/nodes/whatsapp.mdx +305 -0
  147. package/docs-MachinaOs/quickstart.mdx +119 -0
  148. package/docs-MachinaOs/tutorials/ai-agent-workflow.mdx +177 -0
  149. package/docs-MachinaOs/tutorials/android-automation.mdx +242 -0
  150. package/docs-MachinaOs/tutorials/first-workflow.mdx +134 -0
  151. package/docs-MachinaOs/tutorials/whatsapp-automation.mdx +185 -0
  152. package/nul +0 -0
  153. package/package.json +70 -0
  154. package/scripts/build.js +158 -0
  155. package/scripts/check-ports.ps1 +33 -0
  156. package/scripts/clean.js +40 -0
  157. package/scripts/docker.js +93 -0
  158. package/scripts/kill-port.ps1 +154 -0
  159. package/scripts/start.js +210 -0
  160. package/scripts/stop.js +325 -0
  161. package/server/.dockerignore +44 -0
  162. package/server/Dockerfile +45 -0
  163. package/server/constants.py +249 -0
  164. package/server/core/__init__.py +1 -0
  165. package/server/core/cache.py +461 -0
  166. package/server/core/config.py +128 -0
  167. package/server/core/container.py +99 -0
  168. package/server/core/database.py +1211 -0
  169. package/server/core/logging.py +314 -0
  170. package/server/main.py +289 -0
  171. package/server/middleware/__init__.py +5 -0
  172. package/server/middleware/auth.py +89 -0
  173. package/server/models/__init__.py +1 -0
  174. package/server/models/auth.py +52 -0
  175. package/server/models/cache.py +24 -0
  176. package/server/models/database.py +211 -0
  177. package/server/models/nodes.py +455 -0
  178. package/server/package.json +9 -0
  179. package/server/pyproject.toml +72 -0
  180. package/server/requirements.txt +83 -0
  181. package/server/routers/__init__.py +1 -0
  182. package/server/routers/android.py +294 -0
  183. package/server/routers/auth.py +203 -0
  184. package/server/routers/database.py +151 -0
  185. package/server/routers/maps.py +142 -0
  186. package/server/routers/nodejs_compat.py +289 -0
  187. package/server/routers/webhook.py +90 -0
  188. package/server/routers/websocket.py +2127 -0
  189. package/server/routers/whatsapp.py +761 -0
  190. package/server/routers/workflow.py +200 -0
  191. package/server/services/__init__.py +1 -0
  192. package/server/services/ai.py +2415 -0
  193. package/server/services/android/__init__.py +27 -0
  194. package/server/services/android/broadcaster.py +114 -0
  195. package/server/services/android/client.py +608 -0
  196. package/server/services/android/manager.py +78 -0
  197. package/server/services/android/protocol.py +165 -0
  198. package/server/services/android_service.py +588 -0
  199. package/server/services/auth.py +131 -0
  200. package/server/services/chat_client.py +160 -0
  201. package/server/services/deployment/__init__.py +12 -0
  202. package/server/services/deployment/manager.py +706 -0
  203. package/server/services/deployment/state.py +47 -0
  204. package/server/services/deployment/triggers.py +275 -0
  205. package/server/services/event_waiter.py +785 -0
  206. package/server/services/execution/__init__.py +77 -0
  207. package/server/services/execution/cache.py +769 -0
  208. package/server/services/execution/conditions.py +373 -0
  209. package/server/services/execution/dlq.py +132 -0
  210. package/server/services/execution/executor.py +1351 -0
  211. package/server/services/execution/models.py +531 -0
  212. package/server/services/execution/recovery.py +235 -0
  213. package/server/services/handlers/__init__.py +126 -0
  214. package/server/services/handlers/ai.py +355 -0
  215. package/server/services/handlers/android.py +260 -0
  216. package/server/services/handlers/code.py +278 -0
  217. package/server/services/handlers/document.py +598 -0
  218. package/server/services/handlers/http.py +193 -0
  219. package/server/services/handlers/polyglot.py +105 -0
  220. package/server/services/handlers/tools.py +845 -0
  221. package/server/services/handlers/triggers.py +107 -0
  222. package/server/services/handlers/utility.py +822 -0
  223. package/server/services/handlers/whatsapp.py +476 -0
  224. package/server/services/maps.py +289 -0
  225. package/server/services/memory_store.py +103 -0
  226. package/server/services/node_executor.py +375 -0
  227. package/server/services/parameter_resolver.py +218 -0
  228. package/server/services/polyglot_client.py +169 -0
  229. package/server/services/scheduler.py +155 -0
  230. package/server/services/skill_loader.py +417 -0
  231. package/server/services/status_broadcaster.py +826 -0
  232. package/server/services/temporal/__init__.py +23 -0
  233. package/server/services/temporal/activities.py +344 -0
  234. package/server/services/temporal/client.py +76 -0
  235. package/server/services/temporal/executor.py +147 -0
  236. package/server/services/temporal/worker.py +251 -0
  237. package/server/services/temporal/workflow.py +355 -0
  238. package/server/services/temporal/ws_client.py +236 -0
  239. package/server/services/text.py +111 -0
  240. package/server/services/user_auth.py +172 -0
  241. package/server/services/websocket_client.py +29 -0
  242. package/server/services/workflow.py +597 -0
  243. package/server/skills/android-skill/SKILL.md +82 -0
  244. package/server/skills/assistant-personality/SKILL.md +45 -0
  245. package/server/skills/code-skill/SKILL.md +140 -0
  246. package/server/skills/http-skill/SKILL.md +161 -0
  247. package/server/skills/maps-skill/SKILL.md +170 -0
  248. package/server/skills/memory-skill/SKILL.md +154 -0
  249. package/server/skills/scheduler-skill/SKILL.md +84 -0
  250. package/server/skills/whatsapp-skill/SKILL.md +283 -0
  251. package/server/uv.lock +2916 -0
  252. package/server/whatsapp-rpc/.dockerignore +30 -0
  253. package/server/whatsapp-rpc/Dockerfile +44 -0
  254. package/server/whatsapp-rpc/Dockerfile.web +17 -0
  255. package/server/whatsapp-rpc/README.md +139 -0
  256. package/server/whatsapp-rpc/cli.js +95 -0
  257. package/server/whatsapp-rpc/configs/config.yaml +7 -0
  258. package/server/whatsapp-rpc/docker-compose.yml +35 -0
  259. package/server/whatsapp-rpc/docs/API.md +410 -0
  260. package/server/whatsapp-rpc/go.mod +67 -0
  261. package/server/whatsapp-rpc/go.sum +203 -0
  262. package/server/whatsapp-rpc/package.json +30 -0
  263. package/server/whatsapp-rpc/schema.json +1294 -0
  264. package/server/whatsapp-rpc/scripts/clean.cjs +66 -0
  265. package/server/whatsapp-rpc/scripts/cli.js +162 -0
  266. package/server/whatsapp-rpc/src/go/cmd/server/main.go +91 -0
  267. package/server/whatsapp-rpc/src/go/config/config.go +49 -0
  268. package/server/whatsapp-rpc/src/go/rpc/rpc.go +446 -0
  269. package/server/whatsapp-rpc/src/go/rpc/server.go +112 -0
  270. package/server/whatsapp-rpc/src/go/whatsapp/history.go +166 -0
  271. package/server/whatsapp-rpc/src/go/whatsapp/messages.go +390 -0
  272. package/server/whatsapp-rpc/src/go/whatsapp/service.go +2130 -0
  273. package/server/whatsapp-rpc/src/go/whatsapp/types.go +261 -0
  274. package/server/whatsapp-rpc/src/python/pyproject.toml +15 -0
  275. package/server/whatsapp-rpc/src/python/whatsapp_rpc/__init__.py +4 -0
  276. package/server/whatsapp-rpc/src/python/whatsapp_rpc/client.py +427 -0
  277. package/server/whatsapp-rpc/web/app.py +609 -0
  278. package/server/whatsapp-rpc/web/requirements.txt +6 -0
  279. package/server/whatsapp-rpc/web/rpc_client.py +427 -0
  280. package/server/whatsapp-rpc/web/static/openapi.yaml +59 -0
  281. package/server/whatsapp-rpc/web/templates/base.html +150 -0
  282. package/server/whatsapp-rpc/web/templates/contacts.html +240 -0
  283. package/server/whatsapp-rpc/web/templates/dashboard.html +320 -0
  284. package/server/whatsapp-rpc/web/templates/groups.html +328 -0
  285. package/server/whatsapp-rpc/web/templates/messages.html +465 -0
  286. package/server/whatsapp-rpc/web/templates/messaging.html +681 -0
  287. package/server/whatsapp-rpc/web/templates/send.html +259 -0
  288. package/server/whatsapp-rpc/web/templates/settings.html +459 -0
@@ -0,0 +1,582 @@
1
+ import { create } from 'zustand';
2
+ import { Node, Edge } from 'reactflow';
3
+ import { SavedWorkflow } from '../components/ui/WorkflowSidebar';
4
+ import { generateWorkflowId } from '../utils/workflow';
5
+ import { theme } from '../styles/theme';
6
+ import {
7
+ exportWorkflowToJSON as exportToJSON,
8
+ exportWorkflowToFile as exportToFile,
9
+ importWorkflowFromJSON as importFromJSON
10
+ } from '../utils/workflowExport';
11
+ import { workflowApi } from '../services/workflowApi';
12
+
13
+ export interface WorkflowData {
14
+ nodes: Node[];
15
+ edges: Edge[];
16
+ name: string;
17
+ id: string;
18
+ createdAt: Date;
19
+ lastModified: Date;
20
+ }
21
+
22
+ // Per-workflow UI state (n8n pattern - each workflow has isolated execution state)
23
+ export interface WorkflowUIState {
24
+ selectedNodeId: string | null;
25
+ executedNodes: string[]; // Array instead of Set for serialization
26
+ executionOrder: string[];
27
+ isExecuting: boolean;
28
+ viewport?: { x: number; y: number; zoom: number };
29
+ }
30
+
31
+ interface AppStore {
32
+ // Core state - SINGLE source of truth
33
+ currentWorkflow: WorkflowData | null;
34
+ hasUnsavedChanges: boolean;
35
+
36
+ // Per-workflow UI state (n8n pattern - isolated per workflow)
37
+ workflowUIStates: Record<string, WorkflowUIState>;
38
+
39
+ // Global UI state (not workflow-specific)
40
+ selectedNode: Node | null; // Kept for backward compatibility, derived from workflowUIStates
41
+ sidebarVisible: boolean;
42
+ componentPaletteVisible: boolean;
43
+ proMode: boolean; // false = noob mode (only AI categories), true = pro mode (all categories)
44
+ whatsappSettingsOpen: boolean;
45
+ androidSettingsOpen: boolean;
46
+ renamingNodeId: string | null;
47
+
48
+ // Saved workflows
49
+ savedWorkflows: SavedWorkflow[];
50
+
51
+ // Workflow actions
52
+ setCurrentWorkflow: (workflow: WorkflowData) => void;
53
+ updateWorkflow: (updates: Partial<Omit<WorkflowData, 'id' | 'createdAt'>>) => void;
54
+ createNewWorkflow: () => void;
55
+ saveWorkflow: () => Promise<void>;
56
+ loadWorkflow: (id: string) => Promise<void>;
57
+ loadSavedWorkflows: () => Promise<void>;
58
+ deleteWorkflow: (id: string) => Promise<boolean>;
59
+ migrateCurrentWorkflow: () => Promise<void>;
60
+
61
+ // UI actions
62
+ setSelectedNode: (node: Node | null) => void;
63
+ toggleSidebar: () => void;
64
+ toggleComponentPalette: () => void;
65
+ toggleProMode: () => void;
66
+ setWhatsAppSettingsOpen: (open: boolean) => void;
67
+ setAndroidSettingsOpen: (open: boolean) => void;
68
+ setRenamingNodeId: (nodeId: string | null) => void;
69
+
70
+ // Per-workflow UI state actions (n8n pattern)
71
+ getWorkflowUIState: (workflowId: string) => WorkflowUIState;
72
+ setWorkflowExecuting: (workflowId: string, isExecuting: boolean) => void;
73
+ setWorkflowExecutedNodes: (workflowId: string, nodes: string[]) => void;
74
+ setWorkflowExecutionOrder: (workflowId: string, order: string[]) => void;
75
+ setWorkflowViewport: (workflowId: string, viewport: { x: number; y: number; zoom: number }) => void;
76
+ clearWorkflowExecutionState: (workflowId: string) => void;
77
+
78
+ // Node/Edge actions (operate on currentWorkflow)
79
+ updateNodeData: (nodeId: string, newData: any) => void;
80
+ updateNodes: (nodes: Node[]) => void;
81
+ updateEdges: (edges: Edge[]) => void;
82
+ addNode: (node: Node) => void;
83
+ removeNodes: (nodeIds: string[]) => void;
84
+ removeEdges: (edgeIds: string[]) => void;
85
+
86
+ // Workflow export/import
87
+ exportWorkflowToJSON: () => string;
88
+ exportWorkflowToFile: () => void;
89
+ importWorkflowFromJSON: (jsonString: string) => void;
90
+ }
91
+
92
+ // Helper functions
93
+ const createDefaultWorkflow = (): WorkflowData => ({
94
+ id: generateWorkflowId(),
95
+ name: theme.constants.defaultWorkflowName,
96
+ nodes: [],
97
+ edges: [],
98
+ createdAt: new Date(),
99
+ lastModified: new Date(),
100
+ });
101
+
102
+ const createDefaultUIState = (): WorkflowUIState => ({
103
+ selectedNodeId: null,
104
+ executedNodes: [],
105
+ executionOrder: [],
106
+ isExecuting: false,
107
+ viewport: undefined,
108
+ });
109
+
110
+ // Helper to migrate old node types
111
+ const migrateNodes = (nodes: Node[]): Node[] => {
112
+ return nodes.map(node => {
113
+ if (node.type === 'googleChatModel') {
114
+ return { ...node, type: 'geminiChatModel' };
115
+ }
116
+ return node;
117
+ });
118
+ };
119
+
120
+ // Storage keys for UI state persistence
121
+ const STORAGE_KEYS = {
122
+ sidebarVisible: 'ui_sidebar_visible',
123
+ componentPaletteVisible: 'ui_component_palette_visible',
124
+ proMode: 'ui_pro_mode',
125
+ };
126
+
127
+ // Helper to load boolean from localStorage
128
+ const loadBooleanFromStorage = (key: string, defaultValue: boolean): boolean => {
129
+ try {
130
+ const saved = localStorage.getItem(key);
131
+ if (saved !== null) {
132
+ return saved === 'true';
133
+ }
134
+ } catch {
135
+ // Ignore storage errors
136
+ }
137
+ return defaultValue;
138
+ };
139
+
140
+ // Helper to save boolean to localStorage
141
+ const saveBooleanToStorage = (key: string, value: boolean): void => {
142
+ try {
143
+ localStorage.setItem(key, String(value));
144
+ } catch {
145
+ // Ignore storage errors
146
+ }
147
+ };
148
+
149
+ export const useAppStore = create<AppStore>((set, get) => ({
150
+ currentWorkflow: null,
151
+ hasUnsavedChanges: false,
152
+ workflowUIStates: {},
153
+ selectedNode: null,
154
+ sidebarVisible: loadBooleanFromStorage(STORAGE_KEYS.sidebarVisible, true),
155
+ componentPaletteVisible: loadBooleanFromStorage(STORAGE_KEYS.componentPaletteVisible, true),
156
+ proMode: loadBooleanFromStorage(STORAGE_KEYS.proMode, false), // Default to noob mode
157
+ whatsappSettingsOpen: false,
158
+ androidSettingsOpen: false,
159
+ renamingNodeId: null,
160
+ savedWorkflows: [],
161
+
162
+ // Workflow management
163
+ setCurrentWorkflow: (workflow) => {
164
+ set({ currentWorkflow: workflow, hasUnsavedChanges: false });
165
+ },
166
+
167
+ updateWorkflow: (updates) => {
168
+ const current = get().currentWorkflow;
169
+ if (!current) return;
170
+
171
+ const updatedWorkflow = {
172
+ ...current,
173
+ ...updates,
174
+ lastModified: new Date(),
175
+ };
176
+
177
+ set({
178
+ currentWorkflow: updatedWorkflow,
179
+ hasUnsavedChanges: true,
180
+ });
181
+ },
182
+
183
+ createNewWorkflow: () => {
184
+ const newWorkflow = createDefaultWorkflow();
185
+ set({
186
+ currentWorkflow: newWorkflow,
187
+ hasUnsavedChanges: false,
188
+ selectedNode: null,
189
+ });
190
+ },
191
+
192
+ saveWorkflow: async () => {
193
+ const { currentWorkflow, savedWorkflows } = get();
194
+ if (!currentWorkflow) return;
195
+
196
+ const updatedWorkflow = {
197
+ ...currentWorkflow,
198
+ lastModified: new Date(),
199
+ };
200
+
201
+ // Save to database
202
+ const success = await workflowApi.saveWorkflow(
203
+ updatedWorkflow.id,
204
+ updatedWorkflow.name,
205
+ { nodes: updatedWorkflow.nodes, edges: updatedWorkflow.edges }
206
+ );
207
+
208
+ if (!success) {
209
+ console.error('Failed to save workflow to database');
210
+ return;
211
+ }
212
+
213
+ // Update local saved workflows list
214
+ const existingIndex = savedWorkflows.findIndex(w => w.id === updatedWorkflow.id);
215
+
216
+ const workflowSummary: SavedWorkflow = {
217
+ id: updatedWorkflow.id,
218
+ name: updatedWorkflow.name,
219
+ createdAt: updatedWorkflow.createdAt,
220
+ lastModified: updatedWorkflow.lastModified,
221
+ nodeCount: updatedWorkflow.nodes.length,
222
+ };
223
+
224
+ let newSavedWorkflows: SavedWorkflow[];
225
+ if (existingIndex >= 0) {
226
+ newSavedWorkflows = [...savedWorkflows];
227
+ newSavedWorkflows[existingIndex] = workflowSummary;
228
+ } else {
229
+ newSavedWorkflows = [workflowSummary, ...savedWorkflows];
230
+ }
231
+
232
+ set({
233
+ currentWorkflow: updatedWorkflow,
234
+ savedWorkflows: newSavedWorkflows,
235
+ hasUnsavedChanges: false,
236
+ });
237
+ },
238
+
239
+ loadWorkflow: async (id) => {
240
+ const result = await workflowApi.getWorkflow(id);
241
+ if (result) {
242
+ // Migrate old node types
243
+ const nodes = migrateNodes(result.data?.nodes || []);
244
+ const edges = result.data?.edges || [];
245
+
246
+ const workflowData: WorkflowData = {
247
+ id: result.id,
248
+ name: result.name,
249
+ nodes,
250
+ edges,
251
+ createdAt: new Date(result.createdAt),
252
+ lastModified: new Date(result.lastModified),
253
+ };
254
+
255
+ set({
256
+ currentWorkflow: workflowData,
257
+ hasUnsavedChanges: false,
258
+ selectedNode: null,
259
+ });
260
+ }
261
+ },
262
+
263
+ loadSavedWorkflows: async () => {
264
+ const workflows = await workflowApi.getAllWorkflows();
265
+ const savedWorkflows: SavedWorkflow[] = workflows.map(w => ({
266
+ id: w.id,
267
+ name: w.name,
268
+ nodeCount: w.nodeCount,
269
+ createdAt: new Date(w.createdAt),
270
+ lastModified: new Date(w.lastModified),
271
+ }));
272
+ set({ savedWorkflows });
273
+ },
274
+
275
+ deleteWorkflow: async (id) => {
276
+ const { currentWorkflow, savedWorkflows } = get();
277
+
278
+ const success = await workflowApi.deleteWorkflow(id);
279
+ if (!success) {
280
+ console.error('Failed to delete workflow from database');
281
+ return false;
282
+ }
283
+
284
+ // Remove from saved workflows list
285
+ const newSavedWorkflows = savedWorkflows.filter(w => w.id !== id);
286
+
287
+ // If deleting current workflow, create a new one
288
+ if (currentWorkflow?.id === id) {
289
+ const newWorkflow = createDefaultWorkflow();
290
+ set({
291
+ currentWorkflow: newWorkflow,
292
+ savedWorkflows: newSavedWorkflows,
293
+ hasUnsavedChanges: false,
294
+ selectedNode: null,
295
+ });
296
+ } else {
297
+ set({ savedWorkflows: newSavedWorkflows });
298
+ }
299
+
300
+ return true;
301
+ },
302
+
303
+ migrateCurrentWorkflow: async () => {
304
+ const { currentWorkflow } = get();
305
+ if (!currentWorkflow || !currentWorkflow.nodes) return;
306
+
307
+ const migratedNodes = migrateNodes(currentWorkflow.nodes);
308
+
309
+ const hasChanges = migratedNodes.some((node, idx) =>
310
+ node.type !== currentWorkflow.nodes[idx]?.type
311
+ );
312
+
313
+ if (hasChanges) {
314
+ const migratedWorkflow = {
315
+ ...currentWorkflow,
316
+ nodes: migratedNodes
317
+ };
318
+
319
+ // Save migrated workflow to database
320
+ await workflowApi.saveWorkflow(
321
+ migratedWorkflow.id,
322
+ migratedWorkflow.name,
323
+ { nodes: migratedWorkflow.nodes, edges: migratedWorkflow.edges }
324
+ );
325
+
326
+ set({
327
+ currentWorkflow: migratedWorkflow,
328
+ hasUnsavedChanges: false
329
+ });
330
+ }
331
+ },
332
+
333
+ // UI management
334
+ setSelectedNode: (node) => {
335
+ set({ selectedNode: node });
336
+ },
337
+
338
+ toggleSidebar: () => {
339
+ set((state) => {
340
+ const newValue = !state.sidebarVisible;
341
+ saveBooleanToStorage(STORAGE_KEYS.sidebarVisible, newValue);
342
+ return { sidebarVisible: newValue };
343
+ });
344
+ },
345
+
346
+ toggleComponentPalette: () => {
347
+ set((state) => {
348
+ const newValue = !state.componentPaletteVisible;
349
+ saveBooleanToStorage(STORAGE_KEYS.componentPaletteVisible, newValue);
350
+ return { componentPaletteVisible: newValue };
351
+ });
352
+ },
353
+
354
+ toggleProMode: () => {
355
+ set((state) => {
356
+ const newValue = !state.proMode;
357
+ saveBooleanToStorage(STORAGE_KEYS.proMode, newValue);
358
+ return { proMode: newValue };
359
+ });
360
+ },
361
+
362
+ setWhatsAppSettingsOpen: (open) => {
363
+ set({ whatsappSettingsOpen: open });
364
+ },
365
+
366
+ setAndroidSettingsOpen: (open) => {
367
+ set({ androidSettingsOpen: open });
368
+ },
369
+
370
+ setRenamingNodeId: (nodeId) => {
371
+ set({ renamingNodeId: nodeId });
372
+ },
373
+
374
+ // Per-workflow UI state management (n8n pattern - isolated execution state per workflow)
375
+ getWorkflowUIState: (workflowId) => {
376
+ const { workflowUIStates } = get();
377
+ return workflowUIStates[workflowId] || createDefaultUIState();
378
+ },
379
+
380
+ setWorkflowExecuting: (workflowId, isExecuting) => {
381
+ set((state) => {
382
+ const prevState = state.workflowUIStates[workflowId];
383
+ return {
384
+ workflowUIStates: {
385
+ ...state.workflowUIStates,
386
+ [workflowId]: {
387
+ ...(prevState || createDefaultUIState()),
388
+ isExecuting,
389
+ },
390
+ },
391
+ };
392
+ });
393
+ },
394
+
395
+ setWorkflowExecutedNodes: (workflowId, nodes) => {
396
+ set((state) => ({
397
+ workflowUIStates: {
398
+ ...state.workflowUIStates,
399
+ [workflowId]: {
400
+ ...(state.workflowUIStates[workflowId] || createDefaultUIState()),
401
+ executedNodes: nodes,
402
+ },
403
+ },
404
+ }));
405
+ },
406
+
407
+ setWorkflowExecutionOrder: (workflowId, order) => {
408
+ set((state) => ({
409
+ workflowUIStates: {
410
+ ...state.workflowUIStates,
411
+ [workflowId]: {
412
+ ...(state.workflowUIStates[workflowId] || createDefaultUIState()),
413
+ executionOrder: order,
414
+ },
415
+ },
416
+ }));
417
+ },
418
+
419
+ setWorkflowViewport: (workflowId, viewport) => {
420
+ set((state) => ({
421
+ workflowUIStates: {
422
+ ...state.workflowUIStates,
423
+ [workflowId]: {
424
+ ...(state.workflowUIStates[workflowId] || createDefaultUIState()),
425
+ viewport,
426
+ },
427
+ },
428
+ }));
429
+ },
430
+
431
+ clearWorkflowExecutionState: (workflowId) => {
432
+ set((state) => ({
433
+ workflowUIStates: {
434
+ ...state.workflowUIStates,
435
+ [workflowId]: {
436
+ ...(state.workflowUIStates[workflowId] || createDefaultUIState()),
437
+ isExecuting: false,
438
+ executedNodes: [],
439
+ executionOrder: [],
440
+ },
441
+ },
442
+ }));
443
+ },
444
+
445
+ updateNodeData: (nodeId, newData) => {
446
+ const { currentWorkflow, selectedNode } = get();
447
+ if (!currentWorkflow) return;
448
+
449
+ const updatedNodes = currentWorkflow.nodes.map(node =>
450
+ node.id === nodeId ? { ...node, data: { ...node.data, ...newData } } : node
451
+ );
452
+
453
+ const updatedWorkflow = {
454
+ ...currentWorkflow,
455
+ nodes: updatedNodes,
456
+ lastModified: new Date(),
457
+ };
458
+
459
+ set({
460
+ currentWorkflow: updatedWorkflow,
461
+ hasUnsavedChanges: true,
462
+ selectedNode: selectedNode?.id === nodeId ?
463
+ { ...selectedNode, data: { ...selectedNode.data, ...newData } } :
464
+ selectedNode
465
+ });
466
+ },
467
+
468
+ updateNodes: (nodes) => {
469
+ const { currentWorkflow } = get();
470
+ if (!currentWorkflow) return;
471
+
472
+ const updatedWorkflow = {
473
+ ...currentWorkflow,
474
+ nodes,
475
+ lastModified: new Date(),
476
+ };
477
+
478
+ set({
479
+ currentWorkflow: updatedWorkflow,
480
+ hasUnsavedChanges: true,
481
+ });
482
+ },
483
+
484
+ updateEdges: (edges) => {
485
+ const { currentWorkflow } = get();
486
+ if (!currentWorkflow) return;
487
+
488
+ const updatedWorkflow = {
489
+ ...currentWorkflow,
490
+ edges,
491
+ lastModified: new Date(),
492
+ };
493
+
494
+ set({
495
+ currentWorkflow: updatedWorkflow,
496
+ hasUnsavedChanges: true,
497
+ });
498
+ },
499
+
500
+ addNode: (node) => {
501
+ const { currentWorkflow } = get();
502
+ if (!currentWorkflow) return;
503
+
504
+ const updatedWorkflow = {
505
+ ...currentWorkflow,
506
+ nodes: [...currentWorkflow.nodes, node],
507
+ lastModified: new Date(),
508
+ };
509
+
510
+ set({
511
+ currentWorkflow: updatedWorkflow,
512
+ hasUnsavedChanges: true,
513
+ });
514
+ },
515
+
516
+ removeNodes: (nodeIds) => {
517
+ const { currentWorkflow } = get();
518
+ if (!currentWorkflow) return;
519
+
520
+ const updatedNodes = currentWorkflow.nodes.filter(node => !nodeIds.includes(node.id));
521
+ const updatedEdges = currentWorkflow.edges.filter(edge =>
522
+ !nodeIds.includes(edge.source) && !nodeIds.includes(edge.target)
523
+ );
524
+
525
+ const updatedWorkflow = {
526
+ ...currentWorkflow,
527
+ nodes: updatedNodes,
528
+ edges: updatedEdges,
529
+ lastModified: new Date(),
530
+ };
531
+
532
+ set({
533
+ currentWorkflow: updatedWorkflow,
534
+ hasUnsavedChanges: true,
535
+ selectedNode: nodeIds.includes(get().selectedNode?.id || '') ? null : get().selectedNode,
536
+ });
537
+ },
538
+
539
+ removeEdges: (edgeIds) => {
540
+ const { currentWorkflow } = get();
541
+ if (!currentWorkflow) return;
542
+
543
+ const updatedEdges = currentWorkflow.edges.filter(edge => !edgeIds.includes(edge.id));
544
+
545
+ const updatedWorkflow = {
546
+ ...currentWorkflow,
547
+ edges: updatedEdges,
548
+ lastModified: new Date(),
549
+ };
550
+
551
+ set({
552
+ currentWorkflow: updatedWorkflow,
553
+ hasUnsavedChanges: true,
554
+ });
555
+ },
556
+
557
+ exportWorkflowToJSON: () => {
558
+ const { currentWorkflow } = get();
559
+ if (!currentWorkflow) {
560
+ throw new Error('No workflow to export');
561
+ }
562
+
563
+ return exportToJSON(currentWorkflow);
564
+ },
565
+
566
+ exportWorkflowToFile: () => {
567
+ const { currentWorkflow } = get();
568
+ if (!currentWorkflow) {
569
+ throw new Error('No workflow to export');
570
+ }
571
+
572
+ exportToFile(currentWorkflow);
573
+ },
574
+
575
+ importWorkflowFromJSON: (jsonString: string) => {
576
+ const workflow = importFromJSON(jsonString);
577
+ set({
578
+ currentWorkflow: workflow,
579
+ hasUnsavedChanges: true
580
+ });
581
+ },
582
+ }));