flock-core 0.5.0b50__py3-none-any.whl → 0.5.0b51__py3-none-any.whl
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.
Potentially problematic release.
This version of flock-core might be problematic. Click here for more details.
- flock/dashboard/launcher.py +1 -1
- flock/frontend/README.md +678 -0
- flock/frontend/docs/DESIGN_SYSTEM.md +1980 -0
- flock/frontend/index.html +12 -0
- flock/frontend/package-lock.json +4347 -0
- flock/frontend/package.json +48 -0
- flock/frontend/src/App.tsx +79 -0
- flock/frontend/src/__tests__/e2e/critical-scenarios.test.tsx +587 -0
- flock/frontend/src/__tests__/integration/filtering-e2e.test.tsx +387 -0
- flock/frontend/src/__tests__/integration/graph-rendering.test.tsx +640 -0
- flock/frontend/src/__tests__/integration/indexeddb-persistence.test.tsx +699 -0
- flock/frontend/src/components/common/BuildInfo.tsx +39 -0
- flock/frontend/src/components/common/EmptyState.module.css +115 -0
- flock/frontend/src/components/common/EmptyState.tsx +128 -0
- flock/frontend/src/components/common/ErrorBoundary.module.css +169 -0
- flock/frontend/src/components/common/ErrorBoundary.tsx +118 -0
- flock/frontend/src/components/common/KeyboardShortcutsDialog.css +251 -0
- flock/frontend/src/components/common/KeyboardShortcutsDialog.tsx +151 -0
- flock/frontend/src/components/common/LoadingSpinner.module.css +97 -0
- flock/frontend/src/components/common/LoadingSpinner.tsx +29 -0
- flock/frontend/src/components/controls/PublishControl.css +547 -0
- flock/frontend/src/components/controls/PublishControl.test.tsx +543 -0
- flock/frontend/src/components/controls/PublishControl.tsx +432 -0
- flock/frontend/src/components/details/DetailWindowContainer.tsx +62 -0
- flock/frontend/src/components/details/LiveOutputTab.test.tsx +792 -0
- flock/frontend/src/components/details/LiveOutputTab.tsx +220 -0
- flock/frontend/src/components/details/MessageHistoryTab.tsx +299 -0
- flock/frontend/src/components/details/NodeDetailWindow.test.tsx +501 -0
- flock/frontend/src/components/details/NodeDetailWindow.tsx +218 -0
- flock/frontend/src/components/details/RunStatusTab.tsx +307 -0
- flock/frontend/src/components/details/tabs.test.tsx +1015 -0
- flock/frontend/src/components/filters/CorrelationIDFilter.module.css +102 -0
- flock/frontend/src/components/filters/CorrelationIDFilter.test.tsx +197 -0
- flock/frontend/src/components/filters/CorrelationIDFilter.tsx +121 -0
- flock/frontend/src/components/filters/FilterBar.module.css +29 -0
- flock/frontend/src/components/filters/FilterBar.test.tsx +133 -0
- flock/frontend/src/components/filters/FilterBar.tsx +33 -0
- flock/frontend/src/components/filters/FilterPills.module.css +79 -0
- flock/frontend/src/components/filters/FilterPills.test.tsx +173 -0
- flock/frontend/src/components/filters/FilterPills.tsx +67 -0
- flock/frontend/src/components/filters/TimeRangeFilter.module.css +91 -0
- flock/frontend/src/components/filters/TimeRangeFilter.test.tsx +154 -0
- flock/frontend/src/components/filters/TimeRangeFilter.tsx +105 -0
- flock/frontend/src/components/graph/AgentNode.test.tsx +75 -0
- flock/frontend/src/components/graph/AgentNode.tsx +322 -0
- flock/frontend/src/components/graph/GraphCanvas.tsx +406 -0
- flock/frontend/src/components/graph/MessageFlowEdge.tsx +128 -0
- flock/frontend/src/components/graph/MessageNode.test.tsx +62 -0
- flock/frontend/src/components/graph/MessageNode.tsx +116 -0
- flock/frontend/src/components/graph/MiniMap.tsx +47 -0
- flock/frontend/src/components/graph/TransformEdge.tsx +123 -0
- flock/frontend/src/components/layout/DashboardLayout.css +407 -0
- flock/frontend/src/components/layout/DashboardLayout.tsx +300 -0
- flock/frontend/src/components/layout/Header.module.css +88 -0
- flock/frontend/src/components/layout/Header.tsx +52 -0
- flock/frontend/src/components/modules/EventLogModule.test.tsx +401 -0
- flock/frontend/src/components/modules/EventLogModule.tsx +396 -0
- flock/frontend/src/components/modules/EventLogModuleWrapper.tsx +17 -0
- flock/frontend/src/components/modules/ModuleRegistry.test.ts +333 -0
- flock/frontend/src/components/modules/ModuleRegistry.ts +85 -0
- flock/frontend/src/components/modules/ModuleWindow.tsx +155 -0
- flock/frontend/src/components/modules/registerModules.ts +20 -0
- flock/frontend/src/components/settings/AdvancedSettings.tsx +175 -0
- flock/frontend/src/components/settings/AppearanceSettings.tsx +185 -0
- flock/frontend/src/components/settings/GraphSettings.tsx +110 -0
- flock/frontend/src/components/settings/SettingsPanel.css +327 -0
- flock/frontend/src/components/settings/SettingsPanel.tsx +131 -0
- flock/frontend/src/components/settings/ThemeSelector.tsx +298 -0
- flock/frontend/src/hooks/useKeyboardShortcuts.ts +148 -0
- flock/frontend/src/hooks/useModulePersistence.test.ts +442 -0
- flock/frontend/src/hooks/useModulePersistence.ts +154 -0
- flock/frontend/src/hooks/useModules.ts +139 -0
- flock/frontend/src/hooks/usePersistence.ts +139 -0
- flock/frontend/src/main.tsx +13 -0
- flock/frontend/src/services/api.ts +213 -0
- flock/frontend/src/services/indexeddb.test.ts +793 -0
- flock/frontend/src/services/indexeddb.ts +794 -0
- flock/frontend/src/services/layout.test.ts +437 -0
- flock/frontend/src/services/layout.ts +146 -0
- flock/frontend/src/services/themeApplicator.ts +140 -0
- flock/frontend/src/services/themeService.ts +77 -0
- flock/frontend/src/services/websocket.test.ts +595 -0
- flock/frontend/src/services/websocket.ts +685 -0
- flock/frontend/src/store/filterStore.test.ts +242 -0
- flock/frontend/src/store/filterStore.ts +103 -0
- flock/frontend/src/store/graphStore.test.ts +186 -0
- flock/frontend/src/store/graphStore.ts +414 -0
- flock/frontend/src/store/moduleStore.test.ts +253 -0
- flock/frontend/src/store/moduleStore.ts +57 -0
- flock/frontend/src/store/settingsStore.ts +188 -0
- flock/frontend/src/store/streamStore.ts +68 -0
- flock/frontend/src/store/uiStore.test.ts +54 -0
- flock/frontend/src/store/uiStore.ts +110 -0
- flock/frontend/src/store/wsStore.ts +34 -0
- flock/frontend/src/styles/index.css +15 -0
- flock/frontend/src/styles/scrollbar.css +47 -0
- flock/frontend/src/styles/variables.css +488 -0
- flock/frontend/src/test/setup.ts +1 -0
- flock/frontend/src/types/filters.ts +14 -0
- flock/frontend/src/types/graph.ts +55 -0
- flock/frontend/src/types/modules.ts +7 -0
- flock/frontend/src/types/theme.ts +55 -0
- flock/frontend/src/utils/mockData.ts +85 -0
- flock/frontend/src/utils/performance.ts +16 -0
- flock/frontend/src/utils/transforms.test.ts +860 -0
- flock/frontend/src/utils/transforms.ts +323 -0
- flock/frontend/src/vite-env.d.ts +17 -0
- flock/frontend/tsconfig.json +27 -0
- flock/frontend/tsconfig.node.json +11 -0
- flock/frontend/vite.config.ts +25 -0
- flock/frontend/vitest.config.ts +11 -0
- {flock_core-0.5.0b50.dist-info → flock_core-0.5.0b51.dist-info}/METADATA +1 -1
- {flock_core-0.5.0b50.dist-info → flock_core-0.5.0b51.dist-info}/RECORD +116 -6
- {flock_core-0.5.0b50.dist-info → flock_core-0.5.0b51.dist-info}/WHEEL +0 -0
- {flock_core-0.5.0b50.dist-info → flock_core-0.5.0b51.dist-info}/entry_points.txt +0 -0
- {flock_core-0.5.0b50.dist-info → flock_core-0.5.0b51.dist-info}/licenses/LICENSE +0 -0
|
@@ -0,0 +1,442 @@
|
|
|
1
|
+
import { describe, it, expect, beforeEach, vi, afterEach } from 'vitest';
|
|
2
|
+
import { renderHook, waitFor } from '@testing-library/react';
|
|
3
|
+
import { useModulePersistence } from './useModulePersistence';
|
|
4
|
+
import { useModuleStore } from '../store/moduleStore';
|
|
5
|
+
import { indexedDBService } from '../services/indexeddb';
|
|
6
|
+
import type { ModuleInstance } from '../types/modules';
|
|
7
|
+
|
|
8
|
+
// Mock IndexedDB service
|
|
9
|
+
vi.mock('../services/indexeddb', () => ({
|
|
10
|
+
indexedDBService: {
|
|
11
|
+
getAllModuleInstances: vi.fn(),
|
|
12
|
+
saveModuleInstance: vi.fn(),
|
|
13
|
+
deleteModuleInstance: vi.fn(),
|
|
14
|
+
},
|
|
15
|
+
}));
|
|
16
|
+
|
|
17
|
+
describe('useModulePersistence', () => {
|
|
18
|
+
beforeEach(() => {
|
|
19
|
+
vi.clearAllMocks();
|
|
20
|
+
// Reset module store
|
|
21
|
+
useModuleStore.setState({ instances: new Map() });
|
|
22
|
+
});
|
|
23
|
+
|
|
24
|
+
afterEach(() => {
|
|
25
|
+
vi.useRealTimers();
|
|
26
|
+
});
|
|
27
|
+
|
|
28
|
+
describe('Module Loading', () => {
|
|
29
|
+
it('should load saved module instances on mount', async () => {
|
|
30
|
+
const savedInstances = [
|
|
31
|
+
{
|
|
32
|
+
instance_id: 'module-1',
|
|
33
|
+
type: 'eventLog',
|
|
34
|
+
position: { x: 100, y: 200 },
|
|
35
|
+
size: { width: 600, height: 400 },
|
|
36
|
+
visible: true,
|
|
37
|
+
created_at: '2025-10-03T00:00:00Z',
|
|
38
|
+
updated_at: '2025-10-03T00:00:00Z',
|
|
39
|
+
},
|
|
40
|
+
{
|
|
41
|
+
instance_id: 'module-2',
|
|
42
|
+
type: 'eventLog',
|
|
43
|
+
position: { x: 300, y: 400 },
|
|
44
|
+
size: { width: 800, height: 600 },
|
|
45
|
+
visible: true,
|
|
46
|
+
created_at: '2025-10-03T00:00:00Z',
|
|
47
|
+
updated_at: '2025-10-03T00:00:00Z',
|
|
48
|
+
},
|
|
49
|
+
];
|
|
50
|
+
|
|
51
|
+
vi.mocked(indexedDBService.getAllModuleInstances).mockResolvedValue(savedInstances);
|
|
52
|
+
|
|
53
|
+
renderHook(() => useModulePersistence());
|
|
54
|
+
|
|
55
|
+
await waitFor(() => {
|
|
56
|
+
const instances = useModuleStore.getState().instances;
|
|
57
|
+
expect(instances.size).toBe(2);
|
|
58
|
+
expect(instances.get('module-1')).toEqual({
|
|
59
|
+
id: 'module-1',
|
|
60
|
+
type: 'eventLog',
|
|
61
|
+
position: { x: 100, y: 200 },
|
|
62
|
+
size: { width: 600, height: 400 },
|
|
63
|
+
visible: true,
|
|
64
|
+
});
|
|
65
|
+
expect(instances.get('module-2')).toEqual({
|
|
66
|
+
id: 'module-2',
|
|
67
|
+
type: 'eventLog',
|
|
68
|
+
position: { x: 300, y: 400 },
|
|
69
|
+
size: { width: 800, height: 600 },
|
|
70
|
+
visible: true,
|
|
71
|
+
});
|
|
72
|
+
});
|
|
73
|
+
});
|
|
74
|
+
|
|
75
|
+
it('should handle empty module instances on mount', async () => {
|
|
76
|
+
vi.mocked(indexedDBService.getAllModuleInstances).mockResolvedValue([]);
|
|
77
|
+
|
|
78
|
+
renderHook(() => useModulePersistence());
|
|
79
|
+
|
|
80
|
+
await waitFor(() => {
|
|
81
|
+
expect(indexedDBService.getAllModuleInstances).toHaveBeenCalledOnce();
|
|
82
|
+
});
|
|
83
|
+
|
|
84
|
+
const instances = useModuleStore.getState().instances;
|
|
85
|
+
expect(instances.size).toBe(0);
|
|
86
|
+
});
|
|
87
|
+
|
|
88
|
+
it('should handle errors when loading instances', async () => {
|
|
89
|
+
const consoleErrorSpy = vi.spyOn(console, 'error').mockImplementation(() => {});
|
|
90
|
+
vi.mocked(indexedDBService.getAllModuleInstances).mockRejectedValue(
|
|
91
|
+
new Error('Database error')
|
|
92
|
+
);
|
|
93
|
+
|
|
94
|
+
renderHook(() => useModulePersistence());
|
|
95
|
+
|
|
96
|
+
await waitFor(() => {
|
|
97
|
+
expect(consoleErrorSpy).toHaveBeenCalledWith(
|
|
98
|
+
'[ModulePersistence] Failed to load module instances:',
|
|
99
|
+
expect.any(Error)
|
|
100
|
+
);
|
|
101
|
+
});
|
|
102
|
+
|
|
103
|
+
consoleErrorSpy.mockRestore();
|
|
104
|
+
});
|
|
105
|
+
});
|
|
106
|
+
|
|
107
|
+
describe('Module Saving', () => {
|
|
108
|
+
it('should save new module instance when added', async () => {
|
|
109
|
+
vi.mocked(indexedDBService.getAllModuleInstances).mockResolvedValue([]);
|
|
110
|
+
|
|
111
|
+
renderHook(() => useModulePersistence());
|
|
112
|
+
|
|
113
|
+
await waitFor(() => {
|
|
114
|
+
expect(indexedDBService.getAllModuleInstances).toHaveBeenCalled();
|
|
115
|
+
});
|
|
116
|
+
|
|
117
|
+
const newInstance: ModuleInstance = {
|
|
118
|
+
id: 'new-module',
|
|
119
|
+
type: 'eventLog',
|
|
120
|
+
position: { x: 150, y: 250 },
|
|
121
|
+
size: { width: 700, height: 500 },
|
|
122
|
+
visible: true,
|
|
123
|
+
};
|
|
124
|
+
|
|
125
|
+
useModuleStore.getState().addModule(newInstance);
|
|
126
|
+
|
|
127
|
+
// Wait for debounced save (300ms + buffer)
|
|
128
|
+
await new Promise((resolve) => setTimeout(resolve, 400));
|
|
129
|
+
|
|
130
|
+
await waitFor(() => {
|
|
131
|
+
expect(indexedDBService.saveModuleInstance).toHaveBeenCalledWith(
|
|
132
|
+
expect.objectContaining({
|
|
133
|
+
instance_id: 'new-module',
|
|
134
|
+
type: 'eventLog',
|
|
135
|
+
position: { x: 150, y: 250 },
|
|
136
|
+
size: { width: 700, height: 500 },
|
|
137
|
+
visible: true,
|
|
138
|
+
})
|
|
139
|
+
);
|
|
140
|
+
});
|
|
141
|
+
});
|
|
142
|
+
|
|
143
|
+
it('should debounce saves when position changes rapidly', async () => {
|
|
144
|
+
vi.mocked(indexedDBService.getAllModuleInstances).mockResolvedValue([]);
|
|
145
|
+
|
|
146
|
+
renderHook(() => useModulePersistence());
|
|
147
|
+
|
|
148
|
+
await waitFor(() => {
|
|
149
|
+
expect(indexedDBService.getAllModuleInstances).toHaveBeenCalled();
|
|
150
|
+
});
|
|
151
|
+
|
|
152
|
+
const instance: ModuleInstance = {
|
|
153
|
+
id: 'drag-module',
|
|
154
|
+
type: 'eventLog',
|
|
155
|
+
position: { x: 100, y: 100 },
|
|
156
|
+
size: { width: 600, height: 400 },
|
|
157
|
+
visible: true,
|
|
158
|
+
};
|
|
159
|
+
|
|
160
|
+
useModuleStore.getState().addModule(instance);
|
|
161
|
+
|
|
162
|
+
// Wait for initial add to save
|
|
163
|
+
await new Promise((resolve) => setTimeout(resolve, 400));
|
|
164
|
+
|
|
165
|
+
// Rapidly update position (simulating drag)
|
|
166
|
+
useModuleStore.getState().updateModule('drag-module', { position: { x: 110, y: 110 } });
|
|
167
|
+
await new Promise((resolve) => setTimeout(resolve, 100));
|
|
168
|
+
|
|
169
|
+
useModuleStore.getState().updateModule('drag-module', { position: { x: 120, y: 120 } });
|
|
170
|
+
await new Promise((resolve) => setTimeout(resolve, 100));
|
|
171
|
+
|
|
172
|
+
useModuleStore.getState().updateModule('drag-module', { position: { x: 130, y: 130 } });
|
|
173
|
+
|
|
174
|
+
// Wait for debounced save
|
|
175
|
+
await new Promise((resolve) => setTimeout(resolve, 400));
|
|
176
|
+
|
|
177
|
+
await waitFor(() => {
|
|
178
|
+
// Should have saved at least twice: initial add + final position
|
|
179
|
+
expect(indexedDBService.saveModuleInstance).toHaveBeenCalled();
|
|
180
|
+
expect(indexedDBService.saveModuleInstance).toHaveBeenLastCalledWith(
|
|
181
|
+
expect.objectContaining({
|
|
182
|
+
position: { x: 130, y: 130 },
|
|
183
|
+
})
|
|
184
|
+
);
|
|
185
|
+
});
|
|
186
|
+
});
|
|
187
|
+
|
|
188
|
+
it('should save when size changes', async () => {
|
|
189
|
+
vi.mocked(indexedDBService.getAllModuleInstances).mockResolvedValue([]);
|
|
190
|
+
|
|
191
|
+
renderHook(() => useModulePersistence());
|
|
192
|
+
|
|
193
|
+
await waitFor(() => {
|
|
194
|
+
expect(indexedDBService.getAllModuleInstances).toHaveBeenCalled();
|
|
195
|
+
});
|
|
196
|
+
|
|
197
|
+
const instance: ModuleInstance = {
|
|
198
|
+
id: 'resize-module',
|
|
199
|
+
type: 'eventLog',
|
|
200
|
+
position: { x: 100, y: 100 },
|
|
201
|
+
size: { width: 600, height: 400 },
|
|
202
|
+
visible: true,
|
|
203
|
+
};
|
|
204
|
+
|
|
205
|
+
useModuleStore.getState().addModule(instance);
|
|
206
|
+
await new Promise((resolve) => setTimeout(resolve, 400));
|
|
207
|
+
|
|
208
|
+
// Update size
|
|
209
|
+
useModuleStore.getState().updateModule('resize-module', {
|
|
210
|
+
size: { width: 800, height: 600 },
|
|
211
|
+
});
|
|
212
|
+
|
|
213
|
+
await new Promise((resolve) => setTimeout(resolve, 400));
|
|
214
|
+
|
|
215
|
+
await waitFor(() => {
|
|
216
|
+
expect(indexedDBService.saveModuleInstance).toHaveBeenCalledWith(
|
|
217
|
+
expect.objectContaining({
|
|
218
|
+
size: { width: 800, height: 600 },
|
|
219
|
+
})
|
|
220
|
+
);
|
|
221
|
+
});
|
|
222
|
+
});
|
|
223
|
+
|
|
224
|
+
it('should save when visibility changes', async () => {
|
|
225
|
+
vi.mocked(indexedDBService.getAllModuleInstances).mockResolvedValue([]);
|
|
226
|
+
|
|
227
|
+
renderHook(() => useModulePersistence());
|
|
228
|
+
|
|
229
|
+
await waitFor(() => {
|
|
230
|
+
expect(indexedDBService.getAllModuleInstances).toHaveBeenCalled();
|
|
231
|
+
});
|
|
232
|
+
|
|
233
|
+
const instance: ModuleInstance = {
|
|
234
|
+
id: 'toggle-module',
|
|
235
|
+
type: 'eventLog',
|
|
236
|
+
position: { x: 100, y: 100 },
|
|
237
|
+
size: { width: 600, height: 400 },
|
|
238
|
+
visible: true,
|
|
239
|
+
};
|
|
240
|
+
|
|
241
|
+
useModuleStore.getState().addModule(instance);
|
|
242
|
+
await new Promise((resolve) => setTimeout(resolve, 400));
|
|
243
|
+
|
|
244
|
+
// Toggle visibility
|
|
245
|
+
useModuleStore.getState().toggleVisibility('toggle-module');
|
|
246
|
+
|
|
247
|
+
await new Promise((resolve) => setTimeout(resolve, 400));
|
|
248
|
+
|
|
249
|
+
await waitFor(() => {
|
|
250
|
+
expect(indexedDBService.saveModuleInstance).toHaveBeenCalledWith(
|
|
251
|
+
expect.objectContaining({
|
|
252
|
+
visible: false,
|
|
253
|
+
})
|
|
254
|
+
);
|
|
255
|
+
});
|
|
256
|
+
});
|
|
257
|
+
|
|
258
|
+
it('should handle save errors gracefully', async () => {
|
|
259
|
+
const consoleErrorSpy = vi.spyOn(console, 'error').mockImplementation(() => {});
|
|
260
|
+
vi.mocked(indexedDBService.getAllModuleInstances).mockResolvedValue([]);
|
|
261
|
+
vi.mocked(indexedDBService.saveModuleInstance).mockRejectedValue(
|
|
262
|
+
new Error('Save failed')
|
|
263
|
+
);
|
|
264
|
+
|
|
265
|
+
renderHook(() => useModulePersistence());
|
|
266
|
+
|
|
267
|
+
await waitFor(() => {
|
|
268
|
+
expect(indexedDBService.getAllModuleInstances).toHaveBeenCalled();
|
|
269
|
+
});
|
|
270
|
+
|
|
271
|
+
const instance: ModuleInstance = {
|
|
272
|
+
id: 'error-module',
|
|
273
|
+
type: 'eventLog',
|
|
274
|
+
position: { x: 100, y: 100 },
|
|
275
|
+
size: { width: 600, height: 400 },
|
|
276
|
+
visible: true,
|
|
277
|
+
};
|
|
278
|
+
|
|
279
|
+
useModuleStore.getState().addModule(instance);
|
|
280
|
+
await new Promise((resolve) => setTimeout(resolve, 400));
|
|
281
|
+
|
|
282
|
+
await waitFor(() => {
|
|
283
|
+
expect(consoleErrorSpy).toHaveBeenCalledWith(
|
|
284
|
+
expect.stringContaining('Failed to save instance'),
|
|
285
|
+
expect.any(Error)
|
|
286
|
+
);
|
|
287
|
+
});
|
|
288
|
+
|
|
289
|
+
consoleErrorSpy.mockRestore();
|
|
290
|
+
});
|
|
291
|
+
});
|
|
292
|
+
|
|
293
|
+
describe('Module Deletion', () => {
|
|
294
|
+
it('should delete module instance when removed', async () => {
|
|
295
|
+
vi.mocked(indexedDBService.getAllModuleInstances).mockResolvedValue([]);
|
|
296
|
+
|
|
297
|
+
renderHook(() => useModulePersistence());
|
|
298
|
+
|
|
299
|
+
await waitFor(() => {
|
|
300
|
+
expect(indexedDBService.getAllModuleInstances).toHaveBeenCalled();
|
|
301
|
+
});
|
|
302
|
+
|
|
303
|
+
const instance: ModuleInstance = {
|
|
304
|
+
id: 'delete-module',
|
|
305
|
+
type: 'eventLog',
|
|
306
|
+
position: { x: 100, y: 100 },
|
|
307
|
+
size: { width: 600, height: 400 },
|
|
308
|
+
visible: true,
|
|
309
|
+
};
|
|
310
|
+
|
|
311
|
+
useModuleStore.getState().addModule(instance);
|
|
312
|
+
await new Promise((resolve) => setTimeout(resolve, 400));
|
|
313
|
+
|
|
314
|
+
// Remove the instance
|
|
315
|
+
useModuleStore.getState().removeModule('delete-module');
|
|
316
|
+
|
|
317
|
+
await waitFor(() => {
|
|
318
|
+
expect(indexedDBService.deleteModuleInstance).toHaveBeenCalledWith('delete-module');
|
|
319
|
+
});
|
|
320
|
+
});
|
|
321
|
+
|
|
322
|
+
it('should cancel pending save when module is deleted', async () => {
|
|
323
|
+
vi.mocked(indexedDBService.getAllModuleInstances).mockResolvedValue([]);
|
|
324
|
+
|
|
325
|
+
renderHook(() => useModulePersistence());
|
|
326
|
+
|
|
327
|
+
await waitFor(() => {
|
|
328
|
+
expect(indexedDBService.getAllModuleInstances).toHaveBeenCalled();
|
|
329
|
+
});
|
|
330
|
+
|
|
331
|
+
const instance: ModuleInstance = {
|
|
332
|
+
id: 'cancel-module',
|
|
333
|
+
type: 'eventLog',
|
|
334
|
+
position: { x: 100, y: 100 },
|
|
335
|
+
size: { width: 600, height: 400 },
|
|
336
|
+
visible: true,
|
|
337
|
+
};
|
|
338
|
+
|
|
339
|
+
useModuleStore.getState().addModule(instance);
|
|
340
|
+
|
|
341
|
+
// Wait for initial add to save
|
|
342
|
+
await new Promise((resolve) => setTimeout(resolve, 400));
|
|
343
|
+
|
|
344
|
+
const initialCallCount = vi.mocked(indexedDBService.saveModuleInstance).mock.calls.length;
|
|
345
|
+
|
|
346
|
+
// Update position (triggers debounced save)
|
|
347
|
+
useModuleStore.getState().updateModule('cancel-module', { position: { x: 200, y: 200 } });
|
|
348
|
+
|
|
349
|
+
// Remove before debounce timer fires (100ms < 300ms debounce)
|
|
350
|
+
await new Promise((resolve) => setTimeout(resolve, 100));
|
|
351
|
+
useModuleStore.getState().removeModule('cancel-module');
|
|
352
|
+
|
|
353
|
+
// Wait for potential saves
|
|
354
|
+
await new Promise((resolve) => setTimeout(resolve, 400));
|
|
355
|
+
|
|
356
|
+
await waitFor(() => {
|
|
357
|
+
expect(indexedDBService.deleteModuleInstance).toHaveBeenCalledWith('cancel-module');
|
|
358
|
+
});
|
|
359
|
+
|
|
360
|
+
// save should not have been called again after the update (timer was cancelled)
|
|
361
|
+
const finalCallCount = vi.mocked(indexedDBService.saveModuleInstance).mock.calls.length;
|
|
362
|
+
expect(finalCallCount).toBe(initialCallCount);
|
|
363
|
+
});
|
|
364
|
+
|
|
365
|
+
it('should handle delete errors gracefully', async () => {
|
|
366
|
+
const consoleErrorSpy = vi.spyOn(console, 'error').mockImplementation(() => {});
|
|
367
|
+
vi.mocked(indexedDBService.getAllModuleInstances).mockResolvedValue([]);
|
|
368
|
+
vi.mocked(indexedDBService.deleteModuleInstance).mockRejectedValue(
|
|
369
|
+
new Error('Delete failed')
|
|
370
|
+
);
|
|
371
|
+
|
|
372
|
+
renderHook(() => useModulePersistence());
|
|
373
|
+
|
|
374
|
+
await waitFor(() => {
|
|
375
|
+
expect(indexedDBService.getAllModuleInstances).toHaveBeenCalled();
|
|
376
|
+
});
|
|
377
|
+
|
|
378
|
+
const instance: ModuleInstance = {
|
|
379
|
+
id: 'error-delete-module',
|
|
380
|
+
type: 'eventLog',
|
|
381
|
+
position: { x: 100, y: 100 },
|
|
382
|
+
size: { width: 600, height: 400 },
|
|
383
|
+
visible: true,
|
|
384
|
+
};
|
|
385
|
+
|
|
386
|
+
useModuleStore.getState().addModule(instance);
|
|
387
|
+
await new Promise((resolve) => setTimeout(resolve, 400));
|
|
388
|
+
|
|
389
|
+
useModuleStore.getState().removeModule('error-delete-module');
|
|
390
|
+
|
|
391
|
+
await waitFor(() => {
|
|
392
|
+
expect(consoleErrorSpy).toHaveBeenCalledWith(
|
|
393
|
+
expect.stringContaining('Failed to delete instance'),
|
|
394
|
+
expect.any(Error)
|
|
395
|
+
);
|
|
396
|
+
});
|
|
397
|
+
|
|
398
|
+
consoleErrorSpy.mockRestore();
|
|
399
|
+
});
|
|
400
|
+
});
|
|
401
|
+
|
|
402
|
+
describe('Timer Cleanup', () => {
|
|
403
|
+
it('should cleanup timers on unmount', async () => {
|
|
404
|
+
vi.mocked(indexedDBService.getAllModuleInstances).mockResolvedValue([]);
|
|
405
|
+
|
|
406
|
+
const { unmount } = renderHook(() => useModulePersistence());
|
|
407
|
+
|
|
408
|
+
await waitFor(() => {
|
|
409
|
+
expect(indexedDBService.getAllModuleInstances).toHaveBeenCalled();
|
|
410
|
+
});
|
|
411
|
+
|
|
412
|
+
const instance: ModuleInstance = {
|
|
413
|
+
id: 'cleanup-module',
|
|
414
|
+
type: 'eventLog',
|
|
415
|
+
position: { x: 100, y: 100 },
|
|
416
|
+
size: { width: 600, height: 400 },
|
|
417
|
+
visible: true,
|
|
418
|
+
};
|
|
419
|
+
|
|
420
|
+
useModuleStore.getState().addModule(instance);
|
|
421
|
+
|
|
422
|
+
// Wait for initial add to potentially save
|
|
423
|
+
await new Promise((resolve) => setTimeout(resolve, 400));
|
|
424
|
+
|
|
425
|
+
const callCountBeforeUpdate = vi.mocked(indexedDBService.saveModuleInstance).mock.calls.length;
|
|
426
|
+
|
|
427
|
+
// Update position (triggers debounced save)
|
|
428
|
+
useModuleStore.getState().updateModule('cleanup-module', { position: { x: 200, y: 200 } });
|
|
429
|
+
|
|
430
|
+
// Unmount before timer fires (100ms < 300ms debounce)
|
|
431
|
+
await new Promise((resolve) => setTimeout(resolve, 100));
|
|
432
|
+
unmount();
|
|
433
|
+
|
|
434
|
+
// Wait for potential timer - save should not be called after unmount
|
|
435
|
+
await new Promise((resolve) => setTimeout(resolve, 400));
|
|
436
|
+
|
|
437
|
+
// The update should NOT have saved because we unmounted before debounce completed
|
|
438
|
+
const finalCallCount = vi.mocked(indexedDBService.saveModuleInstance).mock.calls.length;
|
|
439
|
+
expect(finalCallCount).toBe(callCountBeforeUpdate);
|
|
440
|
+
});
|
|
441
|
+
});
|
|
442
|
+
});
|
|
@@ -0,0 +1,154 @@
|
|
|
1
|
+
import { useEffect, useRef, useCallback } from 'react';
|
|
2
|
+
import { useModuleStore } from '../store/moduleStore';
|
|
3
|
+
import { indexedDBService } from '../services/indexeddb';
|
|
4
|
+
import type { ModuleInstance } from '../types/modules';
|
|
5
|
+
|
|
6
|
+
/**
|
|
7
|
+
* Hook to persist module instances to IndexedDB
|
|
8
|
+
*
|
|
9
|
+
* Features:
|
|
10
|
+
* - Loads all module instances on mount
|
|
11
|
+
* - Saves instances when added/updated (with debouncing for drag/resize)
|
|
12
|
+
* - Deletes instances when removed
|
|
13
|
+
* - Debounce: 300ms (same as node position persistence)
|
|
14
|
+
*/
|
|
15
|
+
export const useModulePersistence = () => {
|
|
16
|
+
const instances = useModuleStore((state) => state.instances);
|
|
17
|
+
const addModule = useModuleStore((state) => state.addModule);
|
|
18
|
+
|
|
19
|
+
// Track previous instances to detect changes
|
|
20
|
+
const previousInstancesRef = useRef<Map<string, ModuleInstance>>(new Map());
|
|
21
|
+
|
|
22
|
+
// Debounce timer for saves
|
|
23
|
+
const saveTimerRef = useRef<{ [key: string]: number }>({});
|
|
24
|
+
|
|
25
|
+
// Load module instances on mount
|
|
26
|
+
useEffect(() => {
|
|
27
|
+
const loadModuleInstances = async () => {
|
|
28
|
+
try {
|
|
29
|
+
const savedInstances = await indexedDBService.getAllModuleInstances();
|
|
30
|
+
|
|
31
|
+
// Restore each module instance to the store
|
|
32
|
+
savedInstances.forEach((record) => {
|
|
33
|
+
const instance: ModuleInstance = {
|
|
34
|
+
id: record.instance_id,
|
|
35
|
+
type: record.type,
|
|
36
|
+
position: record.position,
|
|
37
|
+
size: record.size,
|
|
38
|
+
visible: record.visible,
|
|
39
|
+
};
|
|
40
|
+
addModule(instance);
|
|
41
|
+
});
|
|
42
|
+
|
|
43
|
+
console.log(`[ModulePersistence] Restored ${savedInstances.length} module instances`);
|
|
44
|
+
} catch (error) {
|
|
45
|
+
console.error('[ModulePersistence] Failed to load module instances:', error);
|
|
46
|
+
}
|
|
47
|
+
};
|
|
48
|
+
|
|
49
|
+
loadModuleInstances();
|
|
50
|
+
// eslint-disable-next-line react-hooks/exhaustive-deps
|
|
51
|
+
}, []); // Only run on mount
|
|
52
|
+
|
|
53
|
+
// Save module instances when they change (debounced)
|
|
54
|
+
const saveModuleInstance = useCallback(async (instance: ModuleInstance) => {
|
|
55
|
+
// Clear existing timer for this instance
|
|
56
|
+
if (saveTimerRef.current[instance.id]) {
|
|
57
|
+
clearTimeout(saveTimerRef.current[instance.id]);
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
// Debounce: wait 300ms before saving (handles rapid drag/resize updates)
|
|
61
|
+
saveTimerRef.current[instance.id] = setTimeout(async () => {
|
|
62
|
+
try {
|
|
63
|
+
const now = new Date().toISOString();
|
|
64
|
+
const record = {
|
|
65
|
+
instance_id: instance.id,
|
|
66
|
+
type: instance.type,
|
|
67
|
+
position: instance.position,
|
|
68
|
+
size: instance.size,
|
|
69
|
+
visible: instance.visible,
|
|
70
|
+
created_at: now, // Will be overwritten if updating existing
|
|
71
|
+
updated_at: now,
|
|
72
|
+
};
|
|
73
|
+
|
|
74
|
+
await indexedDBService.saveModuleInstance(record);
|
|
75
|
+
console.log(`[ModulePersistence] Saved instance: ${instance.id}`);
|
|
76
|
+
} catch (error) {
|
|
77
|
+
console.error(`[ModulePersistence] Failed to save instance ${instance.id}:`, error);
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
delete saveTimerRef.current[instance.id];
|
|
81
|
+
}, 300);
|
|
82
|
+
}, []);
|
|
83
|
+
|
|
84
|
+
// Delete module instance from IndexedDB
|
|
85
|
+
const deleteModuleInstance = useCallback(async (instanceId: string) => {
|
|
86
|
+
// Clear any pending save timer
|
|
87
|
+
if (saveTimerRef.current[instanceId]) {
|
|
88
|
+
clearTimeout(saveTimerRef.current[instanceId]);
|
|
89
|
+
delete saveTimerRef.current[instanceId];
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
try {
|
|
93
|
+
await indexedDBService.deleteModuleInstance(instanceId);
|
|
94
|
+
console.log(`[ModulePersistence] Deleted instance: ${instanceId}`);
|
|
95
|
+
} catch (error) {
|
|
96
|
+
console.error(`[ModulePersistence] Failed to delete instance ${instanceId}:`, error);
|
|
97
|
+
}
|
|
98
|
+
}, []);
|
|
99
|
+
|
|
100
|
+
// Detect changes in module instances
|
|
101
|
+
useEffect(() => {
|
|
102
|
+
const currentIds = new Set(instances.keys());
|
|
103
|
+
const previousIds = new Set(previousInstancesRef.current.keys());
|
|
104
|
+
|
|
105
|
+
// Detect added instances
|
|
106
|
+
currentIds.forEach((id) => {
|
|
107
|
+
if (!previousIds.has(id)) {
|
|
108
|
+
const instance = instances.get(id);
|
|
109
|
+
if (instance) {
|
|
110
|
+
console.log(`[ModulePersistence] Detected new instance: ${id}`);
|
|
111
|
+
saveModuleInstance(instance);
|
|
112
|
+
}
|
|
113
|
+
}
|
|
114
|
+
});
|
|
115
|
+
|
|
116
|
+
// Detect updated instances
|
|
117
|
+
instances.forEach((instance, id) => {
|
|
118
|
+
const previous = previousInstancesRef.current.get(id);
|
|
119
|
+
if (previous) {
|
|
120
|
+
// Check if position, size, or visibility changed
|
|
121
|
+
const positionChanged =
|
|
122
|
+
previous.position.x !== instance.position.x ||
|
|
123
|
+
previous.position.y !== instance.position.y;
|
|
124
|
+
const sizeChanged =
|
|
125
|
+
previous.size.width !== instance.size.width ||
|
|
126
|
+
previous.size.height !== instance.size.height;
|
|
127
|
+
const visibilityChanged = previous.visible !== instance.visible;
|
|
128
|
+
|
|
129
|
+
if (positionChanged || sizeChanged || visibilityChanged) {
|
|
130
|
+
console.log(`[ModulePersistence] Detected update to instance: ${id}`);
|
|
131
|
+
saveModuleInstance(instance);
|
|
132
|
+
}
|
|
133
|
+
}
|
|
134
|
+
});
|
|
135
|
+
|
|
136
|
+
// Detect removed instances
|
|
137
|
+
previousIds.forEach((id) => {
|
|
138
|
+
if (!currentIds.has(id)) {
|
|
139
|
+
console.log(`[ModulePersistence] Detected removed instance: ${id}`);
|
|
140
|
+
deleteModuleInstance(id);
|
|
141
|
+
}
|
|
142
|
+
});
|
|
143
|
+
|
|
144
|
+
// Update ref for next comparison
|
|
145
|
+
previousInstancesRef.current = new Map(instances);
|
|
146
|
+
}, [instances, saveModuleInstance, deleteModuleInstance]);
|
|
147
|
+
|
|
148
|
+
// Cleanup timers on unmount
|
|
149
|
+
useEffect(() => {
|
|
150
|
+
return () => {
|
|
151
|
+
Object.values(saveTimerRef.current).forEach(clearTimeout);
|
|
152
|
+
};
|
|
153
|
+
}, []);
|
|
154
|
+
};
|