flock-core 0.5.0b50__py3-none-any.whl → 0.5.0b52__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/helper/cli_helper.py +1 -1
- {flock_core-0.5.0b50.dist-info → flock_core-0.5.0b52.dist-info}/METADATA +1 -1
- {flock_core-0.5.0b50.dist-info → flock_core-0.5.0b52.dist-info}/RECORD +117 -7
- {flock_core-0.5.0b50.dist-info → flock_core-0.5.0b52.dist-info}/WHEEL +0 -0
- {flock_core-0.5.0b50.dist-info → flock_core-0.5.0b52.dist-info}/entry_points.txt +0 -0
- {flock_core-0.5.0b50.dist-info → flock_core-0.5.0b52.dist-info}/licenses/LICENSE +0 -0
|
@@ -0,0 +1,333 @@
|
|
|
1
|
+
import { describe, it, expect, beforeEach, vi } from 'vitest';
|
|
2
|
+
import { moduleRegistry, ModuleDefinition } from './ModuleRegistry';
|
|
3
|
+
|
|
4
|
+
// Mock React component for testing
|
|
5
|
+
const MockComponent = () => null;
|
|
6
|
+
|
|
7
|
+
describe('ModuleRegistry', () => {
|
|
8
|
+
beforeEach(() => {
|
|
9
|
+
// Reset the registry before each test
|
|
10
|
+
// Access the private resetInstance method through the class
|
|
11
|
+
(moduleRegistry.constructor as any).resetInstance();
|
|
12
|
+
|
|
13
|
+
// Clear console spies
|
|
14
|
+
vi.clearAllMocks();
|
|
15
|
+
});
|
|
16
|
+
|
|
17
|
+
describe('Module Registration', () => {
|
|
18
|
+
it('should register a new module successfully', () => {
|
|
19
|
+
const consoleSpy = vi.spyOn(console, 'log');
|
|
20
|
+
|
|
21
|
+
const module: ModuleDefinition = {
|
|
22
|
+
id: 'test-module',
|
|
23
|
+
name: 'Test Module',
|
|
24
|
+
description: 'A test module',
|
|
25
|
+
component: MockComponent,
|
|
26
|
+
};
|
|
27
|
+
|
|
28
|
+
moduleRegistry.register(module);
|
|
29
|
+
|
|
30
|
+
const registered = moduleRegistry.get('test-module');
|
|
31
|
+
expect(registered).toBeDefined();
|
|
32
|
+
expect(registered?.id).toBe('test-module');
|
|
33
|
+
expect(registered?.name).toBe('Test Module');
|
|
34
|
+
expect(registered?.description).toBe('A test module');
|
|
35
|
+
expect(consoleSpy).toHaveBeenCalledWith(
|
|
36
|
+
'Module "Test Module" (test-module) registered successfully.'
|
|
37
|
+
);
|
|
38
|
+
});
|
|
39
|
+
|
|
40
|
+
it('should register a module with all optional properties', () => {
|
|
41
|
+
const onMountMock = vi.fn();
|
|
42
|
+
const onUnmountMock = vi.fn();
|
|
43
|
+
|
|
44
|
+
const module: ModuleDefinition = {
|
|
45
|
+
id: 'full-module',
|
|
46
|
+
name: 'Full Module',
|
|
47
|
+
description: 'Module with all properties',
|
|
48
|
+
icon: 'icon-name',
|
|
49
|
+
component: MockComponent,
|
|
50
|
+
onMount: onMountMock,
|
|
51
|
+
onUnmount: onUnmountMock,
|
|
52
|
+
};
|
|
53
|
+
|
|
54
|
+
moduleRegistry.register(module);
|
|
55
|
+
|
|
56
|
+
const registered = moduleRegistry.get('full-module');
|
|
57
|
+
expect(registered).toBeDefined();
|
|
58
|
+
expect(registered?.icon).toBe('icon-name');
|
|
59
|
+
expect(registered?.onMount).toBe(onMountMock);
|
|
60
|
+
expect(registered?.onUnmount).toBe(onUnmountMock);
|
|
61
|
+
});
|
|
62
|
+
|
|
63
|
+
it('should prevent duplicate registration and warn', () => {
|
|
64
|
+
const consoleWarnSpy = vi.spyOn(console, 'warn');
|
|
65
|
+
|
|
66
|
+
const module: ModuleDefinition = {
|
|
67
|
+
id: 'duplicate-module',
|
|
68
|
+
name: 'Duplicate Module',
|
|
69
|
+
description: 'Will be registered twice',
|
|
70
|
+
component: MockComponent,
|
|
71
|
+
};
|
|
72
|
+
|
|
73
|
+
moduleRegistry.register(module);
|
|
74
|
+
moduleRegistry.register(module);
|
|
75
|
+
|
|
76
|
+
expect(consoleWarnSpy).toHaveBeenCalledWith(
|
|
77
|
+
'Module with id "duplicate-module" is already registered. Skipping registration.'
|
|
78
|
+
);
|
|
79
|
+
|
|
80
|
+
// Should only have one module
|
|
81
|
+
const allModules = moduleRegistry.getAll();
|
|
82
|
+
const duplicates = allModules.filter(m => m.id === 'duplicate-module');
|
|
83
|
+
expect(duplicates).toHaveLength(1);
|
|
84
|
+
});
|
|
85
|
+
});
|
|
86
|
+
|
|
87
|
+
describe('Module Retrieval', () => {
|
|
88
|
+
it('should retrieve a module by ID', () => {
|
|
89
|
+
const module: ModuleDefinition = {
|
|
90
|
+
id: 'retrieve-test',
|
|
91
|
+
name: 'Retrieve Test',
|
|
92
|
+
description: 'Test retrieval',
|
|
93
|
+
component: MockComponent,
|
|
94
|
+
};
|
|
95
|
+
|
|
96
|
+
moduleRegistry.register(module);
|
|
97
|
+
const retrieved = moduleRegistry.get('retrieve-test');
|
|
98
|
+
|
|
99
|
+
expect(retrieved).toBeDefined();
|
|
100
|
+
expect(retrieved?.id).toBe('retrieve-test');
|
|
101
|
+
});
|
|
102
|
+
|
|
103
|
+
it('should return undefined for non-existent module', () => {
|
|
104
|
+
const result = moduleRegistry.get('non-existent-module');
|
|
105
|
+
expect(result).toBeUndefined();
|
|
106
|
+
});
|
|
107
|
+
|
|
108
|
+
it('should get all registered modules', () => {
|
|
109
|
+
const module1: ModuleDefinition = {
|
|
110
|
+
id: 'module-1',
|
|
111
|
+
name: 'Module 1',
|
|
112
|
+
description: 'First module',
|
|
113
|
+
component: MockComponent,
|
|
114
|
+
};
|
|
115
|
+
|
|
116
|
+
const module2: ModuleDefinition = {
|
|
117
|
+
id: 'module-2',
|
|
118
|
+
name: 'Module 2',
|
|
119
|
+
description: 'Second module',
|
|
120
|
+
component: MockComponent,
|
|
121
|
+
};
|
|
122
|
+
|
|
123
|
+
const module3: ModuleDefinition = {
|
|
124
|
+
id: 'module-3',
|
|
125
|
+
name: 'Module 3',
|
|
126
|
+
description: 'Third module',
|
|
127
|
+
component: MockComponent,
|
|
128
|
+
};
|
|
129
|
+
|
|
130
|
+
moduleRegistry.register(module1);
|
|
131
|
+
moduleRegistry.register(module2);
|
|
132
|
+
moduleRegistry.register(module3);
|
|
133
|
+
|
|
134
|
+
const allModules = moduleRegistry.getAll();
|
|
135
|
+
expect(allModules).toHaveLength(3);
|
|
136
|
+
expect(allModules.map(m => m.id)).toContain('module-1');
|
|
137
|
+
expect(allModules.map(m => m.id)).toContain('module-2');
|
|
138
|
+
expect(allModules.map(m => m.id)).toContain('module-3');
|
|
139
|
+
});
|
|
140
|
+
|
|
141
|
+
it('should return empty array when no modules registered', () => {
|
|
142
|
+
const allModules = moduleRegistry.getAll();
|
|
143
|
+
expect(allModules).toHaveLength(0);
|
|
144
|
+
expect(allModules).toEqual([]);
|
|
145
|
+
});
|
|
146
|
+
});
|
|
147
|
+
|
|
148
|
+
describe('Module Unregistration', () => {
|
|
149
|
+
it('should unregister a module successfully', () => {
|
|
150
|
+
const consoleSpy = vi.spyOn(console, 'log');
|
|
151
|
+
|
|
152
|
+
const module: ModuleDefinition = {
|
|
153
|
+
id: 'unregister-test',
|
|
154
|
+
name: 'Unregister Test',
|
|
155
|
+
description: 'Test unregistration',
|
|
156
|
+
component: MockComponent,
|
|
157
|
+
};
|
|
158
|
+
|
|
159
|
+
moduleRegistry.register(module);
|
|
160
|
+
expect(moduleRegistry.get('unregister-test')).toBeDefined();
|
|
161
|
+
|
|
162
|
+
moduleRegistry.unregister('unregister-test');
|
|
163
|
+
|
|
164
|
+
expect(moduleRegistry.get('unregister-test')).toBeUndefined();
|
|
165
|
+
expect(consoleSpy).toHaveBeenCalledWith(
|
|
166
|
+
'Module "Unregister Test" (unregister-test) unregistered successfully.'
|
|
167
|
+
);
|
|
168
|
+
});
|
|
169
|
+
|
|
170
|
+
it('should handle unregistering non-existent module gracefully', () => {
|
|
171
|
+
const consoleSpy = vi.spyOn(console, 'log');
|
|
172
|
+
|
|
173
|
+
moduleRegistry.unregister('non-existent');
|
|
174
|
+
|
|
175
|
+
// Should not throw error and should not log anything
|
|
176
|
+
expect(consoleSpy).not.toHaveBeenCalled();
|
|
177
|
+
});
|
|
178
|
+
|
|
179
|
+
it('should remove module from getAll after unregistration', () => {
|
|
180
|
+
const module1: ModuleDefinition = {
|
|
181
|
+
id: 'module-1',
|
|
182
|
+
name: 'Module 1',
|
|
183
|
+
description: 'First module',
|
|
184
|
+
component: MockComponent,
|
|
185
|
+
};
|
|
186
|
+
|
|
187
|
+
const module2: ModuleDefinition = {
|
|
188
|
+
id: 'module-2',
|
|
189
|
+
name: 'Module 2',
|
|
190
|
+
description: 'Second module',
|
|
191
|
+
component: MockComponent,
|
|
192
|
+
};
|
|
193
|
+
|
|
194
|
+
moduleRegistry.register(module1);
|
|
195
|
+
moduleRegistry.register(module2);
|
|
196
|
+
expect(moduleRegistry.getAll()).toHaveLength(2);
|
|
197
|
+
|
|
198
|
+
moduleRegistry.unregister('module-1');
|
|
199
|
+
|
|
200
|
+
const remaining = moduleRegistry.getAll();
|
|
201
|
+
expect(remaining).toHaveLength(1);
|
|
202
|
+
expect(remaining[0]?.id).toBe('module-2');
|
|
203
|
+
});
|
|
204
|
+
});
|
|
205
|
+
|
|
206
|
+
describe('Singleton Pattern', () => {
|
|
207
|
+
it('should maintain singleton instance across method calls', () => {
|
|
208
|
+
const module1: ModuleDefinition = {
|
|
209
|
+
id: 'singleton-test-1',
|
|
210
|
+
name: 'Singleton Test 1',
|
|
211
|
+
description: 'Test singleton behavior',
|
|
212
|
+
component: MockComponent,
|
|
213
|
+
};
|
|
214
|
+
|
|
215
|
+
const module2: ModuleDefinition = {
|
|
216
|
+
id: 'singleton-test-2',
|
|
217
|
+
name: 'Singleton Test 2',
|
|
218
|
+
description: 'Test singleton behavior',
|
|
219
|
+
component: MockComponent,
|
|
220
|
+
};
|
|
221
|
+
|
|
222
|
+
moduleRegistry.register(module1);
|
|
223
|
+
moduleRegistry.register(module2);
|
|
224
|
+
|
|
225
|
+
// Should maintain state across different method calls
|
|
226
|
+
const retrieved1 = moduleRegistry.get('singleton-test-1');
|
|
227
|
+
const retrieved2 = moduleRegistry.get('singleton-test-2');
|
|
228
|
+
const allModules = moduleRegistry.getAll();
|
|
229
|
+
|
|
230
|
+
expect(retrieved1).toBeDefined();
|
|
231
|
+
expect(retrieved2).toBeDefined();
|
|
232
|
+
expect(allModules).toHaveLength(2);
|
|
233
|
+
});
|
|
234
|
+
|
|
235
|
+
it('should persist state after unregister and re-register operations', () => {
|
|
236
|
+
const module1: ModuleDefinition = {
|
|
237
|
+
id: 'persist-1',
|
|
238
|
+
name: 'Persist 1',
|
|
239
|
+
description: 'First persistent module',
|
|
240
|
+
component: MockComponent,
|
|
241
|
+
};
|
|
242
|
+
|
|
243
|
+
const module2: ModuleDefinition = {
|
|
244
|
+
id: 'persist-2',
|
|
245
|
+
name: 'Persist 2',
|
|
246
|
+
description: 'Second persistent module',
|
|
247
|
+
component: MockComponent,
|
|
248
|
+
};
|
|
249
|
+
|
|
250
|
+
const module3: ModuleDefinition = {
|
|
251
|
+
id: 'persist-3',
|
|
252
|
+
name: 'Persist 3',
|
|
253
|
+
description: 'Third persistent module',
|
|
254
|
+
component: MockComponent,
|
|
255
|
+
};
|
|
256
|
+
|
|
257
|
+
// Register three modules
|
|
258
|
+
moduleRegistry.register(module1);
|
|
259
|
+
moduleRegistry.register(module2);
|
|
260
|
+
moduleRegistry.register(module3);
|
|
261
|
+
expect(moduleRegistry.getAll()).toHaveLength(3);
|
|
262
|
+
|
|
263
|
+
// Unregister one
|
|
264
|
+
moduleRegistry.unregister('persist-2');
|
|
265
|
+
expect(moduleRegistry.getAll()).toHaveLength(2);
|
|
266
|
+
|
|
267
|
+
// State should persist - the remaining modules should still be there
|
|
268
|
+
expect(moduleRegistry.get('persist-1')).toBeDefined();
|
|
269
|
+
expect(moduleRegistry.get('persist-2')).toBeUndefined();
|
|
270
|
+
expect(moduleRegistry.get('persist-3')).toBeDefined();
|
|
271
|
+
});
|
|
272
|
+
});
|
|
273
|
+
|
|
274
|
+
describe('Console Logging', () => {
|
|
275
|
+
it('should log registration events with correct format', () => {
|
|
276
|
+
const consoleSpy = vi.spyOn(console, 'log');
|
|
277
|
+
|
|
278
|
+
const module: ModuleDefinition = {
|
|
279
|
+
id: 'log-test',
|
|
280
|
+
name: 'Log Test Module',
|
|
281
|
+
description: 'Test logging',
|
|
282
|
+
component: MockComponent,
|
|
283
|
+
};
|
|
284
|
+
|
|
285
|
+
moduleRegistry.register(module);
|
|
286
|
+
|
|
287
|
+
expect(consoleSpy).toHaveBeenCalledTimes(1);
|
|
288
|
+
expect(consoleSpy).toHaveBeenCalledWith(
|
|
289
|
+
'Module "Log Test Module" (log-test) registered successfully.'
|
|
290
|
+
);
|
|
291
|
+
});
|
|
292
|
+
|
|
293
|
+
it('should log unregistration events with correct format', () => {
|
|
294
|
+
const consoleSpy = vi.spyOn(console, 'log');
|
|
295
|
+
|
|
296
|
+
const module: ModuleDefinition = {
|
|
297
|
+
id: 'unlog-test',
|
|
298
|
+
name: 'Unlog Test Module',
|
|
299
|
+
description: 'Test unregistration logging',
|
|
300
|
+
component: MockComponent,
|
|
301
|
+
};
|
|
302
|
+
|
|
303
|
+
moduleRegistry.register(module);
|
|
304
|
+
consoleSpy.mockClear(); // Clear registration log
|
|
305
|
+
|
|
306
|
+
moduleRegistry.unregister('unlog-test');
|
|
307
|
+
|
|
308
|
+
expect(consoleSpy).toHaveBeenCalledTimes(1);
|
|
309
|
+
expect(consoleSpy).toHaveBeenCalledWith(
|
|
310
|
+
'Module "Unlog Test Module" (unlog-test) unregistered successfully.'
|
|
311
|
+
);
|
|
312
|
+
});
|
|
313
|
+
|
|
314
|
+
it('should warn when registering duplicate module', () => {
|
|
315
|
+
const consoleWarnSpy = vi.spyOn(console, 'warn');
|
|
316
|
+
|
|
317
|
+
const module: ModuleDefinition = {
|
|
318
|
+
id: 'warn-test',
|
|
319
|
+
name: 'Warn Test',
|
|
320
|
+
description: 'Test warning',
|
|
321
|
+
component: MockComponent,
|
|
322
|
+
};
|
|
323
|
+
|
|
324
|
+
moduleRegistry.register(module);
|
|
325
|
+
moduleRegistry.register(module);
|
|
326
|
+
|
|
327
|
+
expect(consoleWarnSpy).toHaveBeenCalledTimes(1);
|
|
328
|
+
expect(consoleWarnSpy).toHaveBeenCalledWith(
|
|
329
|
+
'Module with id "warn-test" is already registered. Skipping registration.'
|
|
330
|
+
);
|
|
331
|
+
});
|
|
332
|
+
});
|
|
333
|
+
});
|
|
@@ -0,0 +1,85 @@
|
|
|
1
|
+
import React from 'react';
|
|
2
|
+
import type { Agent, Message } from '../../types/graph';
|
|
3
|
+
import type { TimeRange } from '../../types/filters';
|
|
4
|
+
|
|
5
|
+
export interface ModuleContext {
|
|
6
|
+
// Data access
|
|
7
|
+
agents: Map<string, Agent>;
|
|
8
|
+
messages: Map<string, Message>;
|
|
9
|
+
events: Message[];
|
|
10
|
+
|
|
11
|
+
// Filter state
|
|
12
|
+
filters: {
|
|
13
|
+
correlationId: string | null;
|
|
14
|
+
timeRange: TimeRange;
|
|
15
|
+
};
|
|
16
|
+
|
|
17
|
+
// Actions
|
|
18
|
+
publish: (artifact: any) => void;
|
|
19
|
+
invoke: (agentName: string, inputs: any[]) => void;
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
export interface ModuleDefinition {
|
|
23
|
+
id: string;
|
|
24
|
+
name: string;
|
|
25
|
+
description: string;
|
|
26
|
+
icon?: string;
|
|
27
|
+
component: React.ComponentType<{ context: ModuleContext }>;
|
|
28
|
+
onMount?: (context: ModuleContext) => void;
|
|
29
|
+
onUnmount?: () => void;
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
export interface ModuleRegistry {
|
|
33
|
+
register: (module: ModuleDefinition) => void;
|
|
34
|
+
unregister: (moduleId: string) => void;
|
|
35
|
+
getAll: () => ModuleDefinition[];
|
|
36
|
+
get: (moduleId: string) => ModuleDefinition | undefined;
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
class ModuleRegistryImpl implements ModuleRegistry {
|
|
40
|
+
private static instance: ModuleRegistryImpl;
|
|
41
|
+
private modules: Map<string, ModuleDefinition> = new Map();
|
|
42
|
+
|
|
43
|
+
private constructor() {}
|
|
44
|
+
|
|
45
|
+
public static getInstance(): ModuleRegistryImpl {
|
|
46
|
+
if (!ModuleRegistryImpl.instance) {
|
|
47
|
+
ModuleRegistryImpl.instance = new ModuleRegistryImpl();
|
|
48
|
+
}
|
|
49
|
+
return ModuleRegistryImpl.instance;
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
public register(module: ModuleDefinition): void {
|
|
53
|
+
if (this.modules.has(module.id)) {
|
|
54
|
+
console.warn(`Module with id "${module.id}" is already registered. Skipping registration.`);
|
|
55
|
+
return;
|
|
56
|
+
}
|
|
57
|
+
this.modules.set(module.id, module);
|
|
58
|
+
console.log(`Module "${module.name}" (${module.id}) registered successfully.`);
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
public unregister(moduleId: string): void {
|
|
62
|
+
if (this.modules.has(moduleId)) {
|
|
63
|
+
const module = this.modules.get(moduleId);
|
|
64
|
+
this.modules.delete(moduleId);
|
|
65
|
+
console.log(`Module "${module?.name}" (${moduleId}) unregistered successfully.`);
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
public getAll(): ModuleDefinition[] {
|
|
70
|
+
return Array.from(this.modules.values());
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
public get(moduleId: string): ModuleDefinition | undefined {
|
|
74
|
+
return this.modules.get(moduleId);
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
// For testing purposes - reset the singleton instance
|
|
78
|
+
public static resetInstance(): void {
|
|
79
|
+
if (ModuleRegistryImpl.instance) {
|
|
80
|
+
ModuleRegistryImpl.instance.modules.clear();
|
|
81
|
+
}
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
export const moduleRegistry = ModuleRegistryImpl.getInstance();
|
|
@@ -0,0 +1,155 @@
|
|
|
1
|
+
import React, { useCallback, memo } from 'react';
|
|
2
|
+
import { Rnd } from 'react-rnd';
|
|
3
|
+
import { useModuleStore } from '../../store/moduleStore';
|
|
4
|
+
import { moduleRegistry } from './ModuleRegistry';
|
|
5
|
+
import { useModules } from '../../hooks/useModules';
|
|
6
|
+
|
|
7
|
+
interface ModuleWindowProps {
|
|
8
|
+
instanceId: string;
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
const ModuleWindow: React.FC<ModuleWindowProps> = memo(({ instanceId }) => {
|
|
12
|
+
const instance = useModuleStore((state) => state.instances.get(instanceId));
|
|
13
|
+
const updateModule = useModuleStore((state) => state.updateModule);
|
|
14
|
+
const removeModule = useModuleStore((state) => state.removeModule);
|
|
15
|
+
|
|
16
|
+
// Get module context from useModules hook
|
|
17
|
+
const { context } = useModules();
|
|
18
|
+
|
|
19
|
+
const handleClose = useCallback(() => {
|
|
20
|
+
removeModule(instanceId);
|
|
21
|
+
}, [instanceId, removeModule]);
|
|
22
|
+
|
|
23
|
+
// Don't render if instance doesn't exist or is not visible
|
|
24
|
+
if (!instance || !instance.visible) return null;
|
|
25
|
+
|
|
26
|
+
// Get module definition from registry
|
|
27
|
+
const moduleDefinition = moduleRegistry.get(instance.type);
|
|
28
|
+
if (!moduleDefinition) {
|
|
29
|
+
console.error(`Module definition not found for type: ${instance.type}`);
|
|
30
|
+
return null;
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
const { position, size } = instance;
|
|
34
|
+
const ModuleComponent = moduleDefinition.component;
|
|
35
|
+
|
|
36
|
+
return (
|
|
37
|
+
<Rnd
|
|
38
|
+
position={position}
|
|
39
|
+
size={size}
|
|
40
|
+
onDragStop={(_e, d) => {
|
|
41
|
+
updateModule(instanceId, {
|
|
42
|
+
position: { x: d.x, y: d.y },
|
|
43
|
+
});
|
|
44
|
+
}}
|
|
45
|
+
onResizeStop={(_e, _direction, ref, _delta, position) => {
|
|
46
|
+
updateModule(instanceId, {
|
|
47
|
+
size: {
|
|
48
|
+
width: parseInt(ref.style.width, 10),
|
|
49
|
+
height: parseInt(ref.style.height, 10),
|
|
50
|
+
},
|
|
51
|
+
position,
|
|
52
|
+
});
|
|
53
|
+
}}
|
|
54
|
+
minWidth={1000}
|
|
55
|
+
minHeight={500}
|
|
56
|
+
bounds="parent"
|
|
57
|
+
dragHandleClassName="module-window-header"
|
|
58
|
+
style={{
|
|
59
|
+
zIndex: 1000,
|
|
60
|
+
display: 'flex',
|
|
61
|
+
flexDirection: 'column',
|
|
62
|
+
}}
|
|
63
|
+
>
|
|
64
|
+
<div
|
|
65
|
+
style={{
|
|
66
|
+
display: 'flex',
|
|
67
|
+
flexDirection: 'column',
|
|
68
|
+
width: '100%',
|
|
69
|
+
height: '100%',
|
|
70
|
+
background: 'var(--color-glass-bg)',
|
|
71
|
+
border: 'var(--border-width-1) solid var(--color-glass-border)',
|
|
72
|
+
borderRadius: 'var(--radius-xl)',
|
|
73
|
+
overflow: 'hidden',
|
|
74
|
+
boxShadow: 'var(--shadow-xl)',
|
|
75
|
+
backdropFilter: 'blur(var(--blur-lg))',
|
|
76
|
+
WebkitBackdropFilter: 'blur(var(--blur-lg))',
|
|
77
|
+
}}
|
|
78
|
+
>
|
|
79
|
+
{/* Header */}
|
|
80
|
+
<div
|
|
81
|
+
className="module-window-header"
|
|
82
|
+
style={{
|
|
83
|
+
display: 'flex',
|
|
84
|
+
alignItems: 'center',
|
|
85
|
+
justifyContent: 'space-between',
|
|
86
|
+
padding: 'var(--space-component-md) var(--space-component-lg)',
|
|
87
|
+
background: 'rgba(42, 42, 50, 0.5)',
|
|
88
|
+
borderBottom: 'var(--border-width-1) solid var(--color-border-subtle)',
|
|
89
|
+
cursor: 'move',
|
|
90
|
+
userSelect: 'none',
|
|
91
|
+
}}
|
|
92
|
+
>
|
|
93
|
+
<div style={{ display: 'flex', alignItems: 'center', gap: 'var(--gap-md)' }}>
|
|
94
|
+
{moduleDefinition.icon && (
|
|
95
|
+
<span style={{ fontSize: 16 }}>{moduleDefinition.icon}</span>
|
|
96
|
+
)}
|
|
97
|
+
<span
|
|
98
|
+
style={{
|
|
99
|
+
color: 'var(--color-text-primary)',
|
|
100
|
+
fontSize: 'var(--font-size-body-sm)',
|
|
101
|
+
fontWeight: 'var(--font-weight-semibold)',
|
|
102
|
+
fontFamily: 'var(--font-family-sans)',
|
|
103
|
+
}}
|
|
104
|
+
>
|
|
105
|
+
{moduleDefinition.name}
|
|
106
|
+
</span>
|
|
107
|
+
</div>
|
|
108
|
+
<button
|
|
109
|
+
onClick={handleClose}
|
|
110
|
+
aria-label="Close window"
|
|
111
|
+
style={{
|
|
112
|
+
background: 'transparent',
|
|
113
|
+
border: 'none',
|
|
114
|
+
color: 'var(--color-text-secondary)',
|
|
115
|
+
fontSize: 'var(--font-size-h3)',
|
|
116
|
+
cursor: 'pointer',
|
|
117
|
+
padding: 'var(--spacing-1) var(--spacing-2)',
|
|
118
|
+
lineHeight: 1,
|
|
119
|
+
borderRadius: 'var(--radius-md)',
|
|
120
|
+
transition: 'var(--transition-colors)',
|
|
121
|
+
display: 'flex',
|
|
122
|
+
alignItems: 'center',
|
|
123
|
+
justifyContent: 'center',
|
|
124
|
+
}}
|
|
125
|
+
onMouseEnter={(e) => {
|
|
126
|
+
e.currentTarget.style.color = 'var(--color-error)';
|
|
127
|
+
e.currentTarget.style.background = 'var(--color-error-bg)';
|
|
128
|
+
}}
|
|
129
|
+
onMouseLeave={(e) => {
|
|
130
|
+
e.currentTarget.style.color = 'var(--color-text-secondary)';
|
|
131
|
+
e.currentTarget.style.background = 'transparent';
|
|
132
|
+
}}
|
|
133
|
+
>
|
|
134
|
+
×
|
|
135
|
+
</button>
|
|
136
|
+
</div>
|
|
137
|
+
|
|
138
|
+
{/* Module Content */}
|
|
139
|
+
<div
|
|
140
|
+
style={{
|
|
141
|
+
flex: 1,
|
|
142
|
+
overflow: 'hidden',
|
|
143
|
+
background: 'var(--color-bg-surface)',
|
|
144
|
+
}}
|
|
145
|
+
>
|
|
146
|
+
<ModuleComponent context={context} />
|
|
147
|
+
</div>
|
|
148
|
+
</div>
|
|
149
|
+
</Rnd>
|
|
150
|
+
);
|
|
151
|
+
});
|
|
152
|
+
|
|
153
|
+
ModuleWindow.displayName = 'ModuleWindow';
|
|
154
|
+
|
|
155
|
+
export default ModuleWindow;
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
import { moduleRegistry } from './ModuleRegistry';
|
|
2
|
+
import EventLogModuleWrapper from './EventLogModuleWrapper';
|
|
3
|
+
|
|
4
|
+
/**
|
|
5
|
+
* Register all available modules
|
|
6
|
+
* This should be called during application initialization
|
|
7
|
+
*/
|
|
8
|
+
export function registerModules(): void {
|
|
9
|
+
// Register EventLog module
|
|
10
|
+
moduleRegistry.register({
|
|
11
|
+
id: 'eventLog',
|
|
12
|
+
name: 'Event Log',
|
|
13
|
+
description: 'View and filter system events',
|
|
14
|
+
icon: '📋',
|
|
15
|
+
component: EventLogModuleWrapper,
|
|
16
|
+
});
|
|
17
|
+
|
|
18
|
+
// Future modules can be registered here
|
|
19
|
+
// moduleRegistry.register({ ... });
|
|
20
|
+
}
|