flock-core 0.5.0b71__py3-none-any.whl → 0.5.0b75__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/agent.py +39 -1
- flock/artifacts.py +17 -10
- flock/cli.py +1 -1
- flock/dashboard/__init__.py +2 -0
- flock/dashboard/collector.py +282 -6
- flock/dashboard/events.py +6 -0
- flock/dashboard/graph_builder.py +563 -0
- flock/dashboard/launcher.py +11 -6
- flock/dashboard/models/graph.py +156 -0
- flock/dashboard/service.py +175 -14
- flock/dashboard/static_v2/assets/index-DFRnI_mt.js +111 -0
- flock/dashboard/static_v2/assets/index-fPLNdmp1.css +1 -0
- flock/dashboard/static_v2/index.html +13 -0
- flock/dashboard/websocket.py +2 -2
- flock/engines/dspy_engine.py +27 -8
- flock/frontend/README.md +6 -6
- flock/frontend/src/App.tsx +23 -31
- flock/frontend/src/__tests__/integration/graph-snapshot.test.tsx +647 -0
- flock/frontend/src/components/details/DetailWindowContainer.tsx +13 -17
- flock/frontend/src/components/details/MessageDetailWindow.tsx +439 -0
- flock/frontend/src/components/details/MessageHistoryTab.tsx +128 -53
- flock/frontend/src/components/details/RunStatusTab.tsx +79 -38
- flock/frontend/src/components/graph/AgentNode.test.tsx +3 -1
- flock/frontend/src/components/graph/AgentNode.tsx +8 -6
- flock/frontend/src/components/graph/GraphCanvas.tsx +13 -8
- flock/frontend/src/components/graph/MessageNode.test.tsx +3 -1
- flock/frontend/src/components/graph/MessageNode.tsx +16 -3
- flock/frontend/src/components/layout/DashboardLayout.tsx +12 -9
- flock/frontend/src/components/modules/HistoricalArtifactsModule.tsx +4 -14
- flock/frontend/src/components/modules/ModuleRegistry.ts +5 -3
- flock/frontend/src/hooks/useModules.ts +12 -4
- flock/frontend/src/hooks/usePersistence.ts +5 -3
- flock/frontend/src/services/api.ts +3 -19
- flock/frontend/src/services/graphService.test.ts +330 -0
- flock/frontend/src/services/graphService.ts +75 -0
- flock/frontend/src/services/websocket.ts +104 -268
- flock/frontend/src/store/filterStore.test.ts +89 -1
- flock/frontend/src/store/filterStore.ts +38 -16
- flock/frontend/src/store/graphStore.test.ts +538 -173
- flock/frontend/src/store/graphStore.ts +374 -465
- flock/frontend/src/store/moduleStore.ts +51 -33
- flock/frontend/src/store/uiStore.ts +23 -11
- flock/frontend/src/types/graph.ts +77 -44
- flock/frontend/src/utils/mockData.ts +16 -3
- flock/frontend/vite.config.ts +2 -2
- flock/orchestrator.py +24 -6
- flock/service.py +2 -2
- flock/store.py +169 -4
- flock/themes/darkmatrix.toml +2 -2
- flock/themes/deep.toml +2 -2
- flock/themes/neopolitan.toml +4 -4
- {flock_core-0.5.0b71.dist-info → flock_core-0.5.0b75.dist-info}/METADATA +1 -1
- {flock_core-0.5.0b71.dist-info → flock_core-0.5.0b75.dist-info}/RECORD +56 -53
- flock/frontend/src/__tests__/e2e/critical-scenarios.test.tsx +0 -586
- flock/frontend/src/__tests__/integration/filtering-e2e.test.tsx +0 -391
- flock/frontend/src/__tests__/integration/graph-rendering.test.tsx +0 -640
- flock/frontend/src/services/websocket.test.ts +0 -595
- flock/frontend/src/utils/transforms.test.ts +0 -860
- flock/frontend/src/utils/transforms.ts +0 -323
- {flock_core-0.5.0b71.dist-info → flock_core-0.5.0b75.dist-info}/WHEEL +0 -0
- {flock_core-0.5.0b71.dist-info → flock_core-0.5.0b75.dist-info}/entry_points.txt +0 -0
- {flock_core-0.5.0b71.dist-info → flock_core-0.5.0b75.dist-info}/licenses/LICENSE +0 -0
|
@@ -0,0 +1,647 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Phase 5: Integration Tests for Backend Snapshot Consumption
|
|
3
|
+
*
|
|
4
|
+
* Tests the complete flow from backend API to UI rendering, including:
|
|
5
|
+
* - Initial graph loading from backend
|
|
6
|
+
* - WebSocket-triggered debounced refreshes
|
|
7
|
+
* - Position persistence
|
|
8
|
+
* - Filter application
|
|
9
|
+
* - Error handling
|
|
10
|
+
* - Empty states
|
|
11
|
+
*/
|
|
12
|
+
|
|
13
|
+
import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
|
|
14
|
+
import { act } from '@testing-library/react';
|
|
15
|
+
import { useGraphStore } from '../../store/graphStore';
|
|
16
|
+
import { useFilterStore } from '../../store/filterStore';
|
|
17
|
+
import { GraphSnapshot } from '../../types/graph';
|
|
18
|
+
|
|
19
|
+
// Mock the graph service
|
|
20
|
+
vi.mock('../../services/graphService', () => ({
|
|
21
|
+
fetchGraphSnapshot: vi.fn(),
|
|
22
|
+
mergeNodePositions: vi.fn((backendNodes) => backendNodes),
|
|
23
|
+
overlayWebSocketState: vi.fn((nodes) => nodes),
|
|
24
|
+
}));
|
|
25
|
+
|
|
26
|
+
// Mock IndexedDB for persistence tests
|
|
27
|
+
const mockIndexedDB = {
|
|
28
|
+
open: vi.fn(),
|
|
29
|
+
databases: vi.fn().mockResolvedValue([]),
|
|
30
|
+
};
|
|
31
|
+
globalThis.indexedDB = mockIndexedDB as any;
|
|
32
|
+
|
|
33
|
+
import * as graphService from '../../services/graphService';
|
|
34
|
+
|
|
35
|
+
describe('Graph Snapshot Integration', () => {
|
|
36
|
+
// Sample backend snapshot fixture
|
|
37
|
+
const createMockSnapshot = (overrides?: Partial<GraphSnapshot>): GraphSnapshot => ({
|
|
38
|
+
generatedAt: '2025-10-11T00:00:00Z',
|
|
39
|
+
viewMode: 'agent',
|
|
40
|
+
filters: {
|
|
41
|
+
correlation_id: null,
|
|
42
|
+
time_range: { preset: 'last10min' },
|
|
43
|
+
artifactTypes: [],
|
|
44
|
+
producers: [],
|
|
45
|
+
tags: [],
|
|
46
|
+
visibility: [],
|
|
47
|
+
},
|
|
48
|
+
nodes: [
|
|
49
|
+
{
|
|
50
|
+
id: 'pizza_master',
|
|
51
|
+
type: 'agent',
|
|
52
|
+
data: { name: 'pizza_master', status: 'idle' },
|
|
53
|
+
position: { x: 100, y: 100 },
|
|
54
|
+
hidden: false,
|
|
55
|
+
},
|
|
56
|
+
{
|
|
57
|
+
id: 'topping_picker',
|
|
58
|
+
type: 'agent',
|
|
59
|
+
data: { name: 'topping_picker', status: 'idle' },
|
|
60
|
+
position: { x: 300, y: 200 },
|
|
61
|
+
hidden: false,
|
|
62
|
+
},
|
|
63
|
+
],
|
|
64
|
+
edges: [
|
|
65
|
+
{
|
|
66
|
+
id: 'edge-1',
|
|
67
|
+
source: 'pizza_master',
|
|
68
|
+
target: 'topping_picker',
|
|
69
|
+
type: 'message_flow',
|
|
70
|
+
label: 'Pizza',
|
|
71
|
+
data: {},
|
|
72
|
+
hidden: false,
|
|
73
|
+
},
|
|
74
|
+
],
|
|
75
|
+
statistics: {
|
|
76
|
+
producedByAgent: {
|
|
77
|
+
pizza_master: { total: 5, byType: { Pizza: 5 } },
|
|
78
|
+
},
|
|
79
|
+
consumedByAgent: {
|
|
80
|
+
topping_picker: { total: 5, byType: { Pizza: 5 } },
|
|
81
|
+
},
|
|
82
|
+
artifactSummary: {
|
|
83
|
+
total: 10,
|
|
84
|
+
by_type: { Pizza: 10 },
|
|
85
|
+
by_producer: { pizza_master: 10 },
|
|
86
|
+
by_visibility: { public: 10 },
|
|
87
|
+
tag_counts: {},
|
|
88
|
+
earliest_created_at: '2025-10-11T00:00:00Z',
|
|
89
|
+
latest_created_at: '2025-10-11T00:05:00Z',
|
|
90
|
+
},
|
|
91
|
+
},
|
|
92
|
+
totalArtifacts: 10,
|
|
93
|
+
truncated: false,
|
|
94
|
+
...overrides,
|
|
95
|
+
});
|
|
96
|
+
|
|
97
|
+
beforeEach(() => {
|
|
98
|
+
// Reset all stores before each test
|
|
99
|
+
useGraphStore.setState({
|
|
100
|
+
agentStatus: new Map(),
|
|
101
|
+
streamingTokens: new Map(),
|
|
102
|
+
nodes: [],
|
|
103
|
+
edges: [],
|
|
104
|
+
statistics: null,
|
|
105
|
+
events: [],
|
|
106
|
+
viewMode: 'agent',
|
|
107
|
+
isLoading: false,
|
|
108
|
+
error: null,
|
|
109
|
+
});
|
|
110
|
+
|
|
111
|
+
useFilterStore.setState({
|
|
112
|
+
correlationId: null,
|
|
113
|
+
timeRange: { preset: 'last10min' },
|
|
114
|
+
selectedArtifactTypes: [],
|
|
115
|
+
selectedProducers: [],
|
|
116
|
+
selectedTags: [],
|
|
117
|
+
selectedVisibility: [],
|
|
118
|
+
availableArtifactTypes: [],
|
|
119
|
+
availableProducers: [],
|
|
120
|
+
availableTags: [],
|
|
121
|
+
availableVisibility: [],
|
|
122
|
+
});
|
|
123
|
+
|
|
124
|
+
vi.clearAllMocks();
|
|
125
|
+
});
|
|
126
|
+
|
|
127
|
+
afterEach(() => {
|
|
128
|
+
vi.restoreAllMocks();
|
|
129
|
+
});
|
|
130
|
+
|
|
131
|
+
describe('Test 1: Initial Graph Loading', () => {
|
|
132
|
+
it('should fetch agent view graph from backend on mount', async () => {
|
|
133
|
+
const mockSnapshot = createMockSnapshot();
|
|
134
|
+
vi.mocked(graphService.fetchGraphSnapshot).mockResolvedValue(mockSnapshot);
|
|
135
|
+
|
|
136
|
+
// Trigger graph generation
|
|
137
|
+
await act(async () => {
|
|
138
|
+
await useGraphStore.getState().generateAgentViewGraph();
|
|
139
|
+
});
|
|
140
|
+
|
|
141
|
+
// Verify backend was called with correct request
|
|
142
|
+
expect(graphService.fetchGraphSnapshot).toHaveBeenCalledWith({
|
|
143
|
+
viewMode: 'agent',
|
|
144
|
+
filters: {
|
|
145
|
+
correlation_id: null,
|
|
146
|
+
time_range: { preset: 'last10min' },
|
|
147
|
+
artifactTypes: [],
|
|
148
|
+
producers: [],
|
|
149
|
+
tags: [],
|
|
150
|
+
visibility: [],
|
|
151
|
+
},
|
|
152
|
+
options: { include_statistics: true },
|
|
153
|
+
});
|
|
154
|
+
|
|
155
|
+
// Verify store was updated with snapshot data
|
|
156
|
+
const state = useGraphStore.getState();
|
|
157
|
+
expect(state.nodes).toHaveLength(2);
|
|
158
|
+
expect(state.edges).toHaveLength(1);
|
|
159
|
+
expect(state.statistics).toBeTruthy();
|
|
160
|
+
expect(state.viewMode).toBe('agent');
|
|
161
|
+
});
|
|
162
|
+
|
|
163
|
+
it('should fetch blackboard view graph from backend', async () => {
|
|
164
|
+
const mockSnapshot = createMockSnapshot({ viewMode: 'blackboard' });
|
|
165
|
+
vi.mocked(graphService.fetchGraphSnapshot).mockResolvedValue(mockSnapshot);
|
|
166
|
+
|
|
167
|
+
await act(async () => {
|
|
168
|
+
await useGraphStore.getState().generateBlackboardViewGraph();
|
|
169
|
+
});
|
|
170
|
+
|
|
171
|
+
expect(graphService.fetchGraphSnapshot).toHaveBeenCalledWith({
|
|
172
|
+
viewMode: 'blackboard',
|
|
173
|
+
filters: expect.any(Object),
|
|
174
|
+
options: { include_statistics: true },
|
|
175
|
+
});
|
|
176
|
+
|
|
177
|
+
const state = useGraphStore.getState();
|
|
178
|
+
expect(state.viewMode).toBe('blackboard');
|
|
179
|
+
});
|
|
180
|
+
|
|
181
|
+
it('should call fetchGraphSnapshot only once on initial load', async () => {
|
|
182
|
+
const mockSnapshot = createMockSnapshot();
|
|
183
|
+
vi.mocked(graphService.fetchGraphSnapshot).mockResolvedValue(mockSnapshot);
|
|
184
|
+
|
|
185
|
+
await act(async () => {
|
|
186
|
+
await useGraphStore.getState().generateAgentViewGraph();
|
|
187
|
+
});
|
|
188
|
+
|
|
189
|
+
expect(graphService.fetchGraphSnapshot).toHaveBeenCalledTimes(1);
|
|
190
|
+
});
|
|
191
|
+
});
|
|
192
|
+
|
|
193
|
+
describe('Test 2: WebSocket Event Handling', () => {
|
|
194
|
+
it('should update agent status immediately without backend fetch', () => {
|
|
195
|
+
// Set up initial nodes
|
|
196
|
+
useGraphStore.setState({
|
|
197
|
+
nodes: [
|
|
198
|
+
{
|
|
199
|
+
id: 'pizza_master',
|
|
200
|
+
type: 'agent',
|
|
201
|
+
data: { name: 'pizza_master', status: 'idle' },
|
|
202
|
+
position: { x: 100, y: 100 },
|
|
203
|
+
},
|
|
204
|
+
],
|
|
205
|
+
});
|
|
206
|
+
|
|
207
|
+
// Update status (simulating WebSocket event)
|
|
208
|
+
act(() => {
|
|
209
|
+
useGraphStore.getState().updateAgentStatus('pizza_master', 'running');
|
|
210
|
+
});
|
|
211
|
+
|
|
212
|
+
// Verify status updated immediately
|
|
213
|
+
const state = useGraphStore.getState();
|
|
214
|
+
const node = state.nodes.find((n) => n.id === 'pizza_master');
|
|
215
|
+
expect(node?.data.status).toBe('running');
|
|
216
|
+
|
|
217
|
+
// Verify NO backend fetch happened (fast path)
|
|
218
|
+
expect(graphService.fetchGraphSnapshot).not.toHaveBeenCalled();
|
|
219
|
+
});
|
|
220
|
+
|
|
221
|
+
it('should update streaming tokens without backend fetch', () => {
|
|
222
|
+
useGraphStore.setState({
|
|
223
|
+
nodes: [
|
|
224
|
+
{
|
|
225
|
+
id: 'pizza_master',
|
|
226
|
+
type: 'agent',
|
|
227
|
+
data: { name: 'pizza_master', streamingTokens: [] },
|
|
228
|
+
position: { x: 100, y: 100 },
|
|
229
|
+
},
|
|
230
|
+
],
|
|
231
|
+
});
|
|
232
|
+
|
|
233
|
+
act(() => {
|
|
234
|
+
useGraphStore.getState().updateStreamingTokens('pizza_master', ['token1', 'token2', 'token3']);
|
|
235
|
+
});
|
|
236
|
+
|
|
237
|
+
const state = useGraphStore.getState();
|
|
238
|
+
const node = state.nodes.find((n) => n.id === 'pizza_master');
|
|
239
|
+
expect(node?.data.streamingTokens).toEqual(['token1', 'token2', 'token3']);
|
|
240
|
+
|
|
241
|
+
// No backend fetch for streaming tokens (fast path)
|
|
242
|
+
expect(graphService.fetchGraphSnapshot).not.toHaveBeenCalled();
|
|
243
|
+
});
|
|
244
|
+
|
|
245
|
+
it('should keep only last 6 streaming tokens', () => {
|
|
246
|
+
useGraphStore.setState({
|
|
247
|
+
nodes: [
|
|
248
|
+
{
|
|
249
|
+
id: 'pizza_master',
|
|
250
|
+
type: 'agent',
|
|
251
|
+
data: { streamingTokens: [] },
|
|
252
|
+
position: { x: 100, y: 100 },
|
|
253
|
+
},
|
|
254
|
+
],
|
|
255
|
+
});
|
|
256
|
+
|
|
257
|
+
const tokens = ['t1', 't2', 't3', 't4', 't5', 't6', 't7', 't8'];
|
|
258
|
+
act(() => {
|
|
259
|
+
useGraphStore.getState().updateStreamingTokens('pizza_master', tokens);
|
|
260
|
+
});
|
|
261
|
+
|
|
262
|
+
const state = useGraphStore.getState();
|
|
263
|
+
const node = state.nodes.find((n) => n.id === 'pizza_master');
|
|
264
|
+
expect(node?.data.streamingTokens).toHaveLength(6);
|
|
265
|
+
expect(node?.data.streamingTokens).toEqual(['t3', 't4', 't5', 't6', 't7', 't8']);
|
|
266
|
+
});
|
|
267
|
+
});
|
|
268
|
+
|
|
269
|
+
describe('Test 3: View Refresh', () => {
|
|
270
|
+
it('should call appropriate view generator based on viewMode', async () => {
|
|
271
|
+
const mockSnapshot = createMockSnapshot();
|
|
272
|
+
vi.mocked(graphService.fetchGraphSnapshot).mockResolvedValue(mockSnapshot);
|
|
273
|
+
|
|
274
|
+
// Set agent view mode
|
|
275
|
+
useGraphStore.setState({ viewMode: 'agent' });
|
|
276
|
+
|
|
277
|
+
await act(async () => {
|
|
278
|
+
await useGraphStore.getState().refreshCurrentView();
|
|
279
|
+
});
|
|
280
|
+
|
|
281
|
+
// Should call agent view
|
|
282
|
+
expect(graphService.fetchGraphSnapshot).toHaveBeenCalledWith(
|
|
283
|
+
expect.objectContaining({ viewMode: 'agent' })
|
|
284
|
+
);
|
|
285
|
+
});
|
|
286
|
+
|
|
287
|
+
it('should refresh blackboard view when viewMode is blackboard', async () => {
|
|
288
|
+
const mockSnapshot = createMockSnapshot({ viewMode: 'blackboard' });
|
|
289
|
+
vi.mocked(graphService.fetchGraphSnapshot).mockResolvedValue(mockSnapshot);
|
|
290
|
+
|
|
291
|
+
// Set blackboard view mode
|
|
292
|
+
useGraphStore.setState({ viewMode: 'blackboard' });
|
|
293
|
+
|
|
294
|
+
await act(async () => {
|
|
295
|
+
await useGraphStore.getState().refreshCurrentView();
|
|
296
|
+
});
|
|
297
|
+
|
|
298
|
+
expect(graphService.fetchGraphSnapshot).toHaveBeenCalledWith(
|
|
299
|
+
expect.objectContaining({ viewMode: 'blackboard' })
|
|
300
|
+
);
|
|
301
|
+
});
|
|
302
|
+
|
|
303
|
+
it('should accumulate events from multiple addEvent calls', () => {
|
|
304
|
+
// Simulate multiple message published events
|
|
305
|
+
act(() => {
|
|
306
|
+
useGraphStore.getState().addEvent({
|
|
307
|
+
id: 'msg1',
|
|
308
|
+
type: 'Pizza',
|
|
309
|
+
producedBy: 'pizza_master',
|
|
310
|
+
timestamp: Date.now(),
|
|
311
|
+
} as any);
|
|
312
|
+
|
|
313
|
+
useGraphStore.getState().addEvent({
|
|
314
|
+
id: 'msg2',
|
|
315
|
+
type: 'Pizza',
|
|
316
|
+
producedBy: 'pizza_master',
|
|
317
|
+
timestamp: Date.now(),
|
|
318
|
+
} as any);
|
|
319
|
+
|
|
320
|
+
useGraphStore.getState().addEvent({
|
|
321
|
+
id: 'msg3',
|
|
322
|
+
type: 'Pizza',
|
|
323
|
+
producedBy: 'pizza_master',
|
|
324
|
+
timestamp: Date.now(),
|
|
325
|
+
} as any);
|
|
326
|
+
});
|
|
327
|
+
|
|
328
|
+
// Verify all 3 events in store
|
|
329
|
+
const state = useGraphStore.getState();
|
|
330
|
+
expect(state.events).toHaveLength(3);
|
|
331
|
+
});
|
|
332
|
+
});
|
|
333
|
+
|
|
334
|
+
describe('Test 4: Position Persistence', () => {
|
|
335
|
+
it('should merge saved positions with backend nodes', async () => {
|
|
336
|
+
// Mock persistence to return saved positions
|
|
337
|
+
vi.mocked(graphService.mergeNodePositions).mockImplementation(
|
|
338
|
+
(backendNodes, saved) => {
|
|
339
|
+
return backendNodes.map((node) => ({
|
|
340
|
+
...node,
|
|
341
|
+
position: saved.get(node.id) || node.position,
|
|
342
|
+
}));
|
|
343
|
+
}
|
|
344
|
+
);
|
|
345
|
+
|
|
346
|
+
const mockSnapshot = createMockSnapshot();
|
|
347
|
+
vi.mocked(graphService.fetchGraphSnapshot).mockResolvedValue(mockSnapshot);
|
|
348
|
+
|
|
349
|
+
await act(async () => {
|
|
350
|
+
await useGraphStore.getState().generateAgentViewGraph();
|
|
351
|
+
});
|
|
352
|
+
|
|
353
|
+
// Verify mergeNodePositions was called with saved positions
|
|
354
|
+
expect(graphService.mergeNodePositions).toHaveBeenCalled();
|
|
355
|
+
});
|
|
356
|
+
|
|
357
|
+
it('should update node position when user drags node', () => {
|
|
358
|
+
useGraphStore.setState({
|
|
359
|
+
nodes: [
|
|
360
|
+
{
|
|
361
|
+
id: 'pizza_master',
|
|
362
|
+
type: 'agent',
|
|
363
|
+
data: {},
|
|
364
|
+
position: { x: 100, y: 100 },
|
|
365
|
+
},
|
|
366
|
+
],
|
|
367
|
+
});
|
|
368
|
+
|
|
369
|
+
// Simulate user dragging node to new position
|
|
370
|
+
act(() => {
|
|
371
|
+
useGraphStore.getState().updateNodePosition('pizza_master', { x: 250, y: 350 });
|
|
372
|
+
});
|
|
373
|
+
|
|
374
|
+
// Verify position updated in store
|
|
375
|
+
const state = useGraphStore.getState();
|
|
376
|
+
const node = state.nodes.find((n) => n.id === 'pizza_master');
|
|
377
|
+
expect(node?.position).toEqual({ x: 250, y: 350 });
|
|
378
|
+
});
|
|
379
|
+
});
|
|
380
|
+
|
|
381
|
+
describe('Test 5: Filter Application', () => {
|
|
382
|
+
it('should trigger backend fetch when filters applied', async () => {
|
|
383
|
+
const mockSnapshot = createMockSnapshot();
|
|
384
|
+
vi.mocked(graphService.fetchGraphSnapshot).mockResolvedValue(mockSnapshot);
|
|
385
|
+
|
|
386
|
+
// Update filters using correct method names
|
|
387
|
+
act(() => {
|
|
388
|
+
useFilterStore.getState().setArtifactTypes(['Pizza']);
|
|
389
|
+
useFilterStore.getState().setProducers(['pizza_master']);
|
|
390
|
+
});
|
|
391
|
+
|
|
392
|
+
// Apply filters
|
|
393
|
+
await act(async () => {
|
|
394
|
+
await useFilterStore.getState().applyFilters();
|
|
395
|
+
});
|
|
396
|
+
|
|
397
|
+
// Verify backend called with updated filters
|
|
398
|
+
expect(graphService.fetchGraphSnapshot).toHaveBeenCalledWith({
|
|
399
|
+
viewMode: 'agent',
|
|
400
|
+
filters: {
|
|
401
|
+
correlation_id: null,
|
|
402
|
+
time_range: expect.objectContaining({ preset: 'last10min' }),
|
|
403
|
+
artifactTypes: ['Pizza'],
|
|
404
|
+
producers: ['pizza_master'],
|
|
405
|
+
tags: [],
|
|
406
|
+
visibility: [],
|
|
407
|
+
},
|
|
408
|
+
options: { include_statistics: true },
|
|
409
|
+
});
|
|
410
|
+
});
|
|
411
|
+
|
|
412
|
+
it('should update available facets from backend statistics', async () => {
|
|
413
|
+
const mockSnapshot = createMockSnapshot();
|
|
414
|
+
vi.mocked(graphService.fetchGraphSnapshot).mockResolvedValue(mockSnapshot);
|
|
415
|
+
|
|
416
|
+
await act(async () => {
|
|
417
|
+
await useGraphStore.getState().generateAgentViewGraph();
|
|
418
|
+
});
|
|
419
|
+
|
|
420
|
+
// Verify filter facets updated from backend
|
|
421
|
+
const filterState = useFilterStore.getState();
|
|
422
|
+
expect(filterState.availableArtifactTypes).toContain('Pizza');
|
|
423
|
+
expect(filterState.availableProducers).toContain('pizza_master');
|
|
424
|
+
expect(filterState.availableVisibility).toContain('public');
|
|
425
|
+
});
|
|
426
|
+
});
|
|
427
|
+
|
|
428
|
+
describe('Test 6: Error Handling', () => {
|
|
429
|
+
it('should handle backend API errors gracefully', async () => {
|
|
430
|
+
vi.mocked(graphService.fetchGraphSnapshot).mockRejectedValue(
|
|
431
|
+
new Error('Backend API unavailable')
|
|
432
|
+
);
|
|
433
|
+
|
|
434
|
+
// Verify error is thrown
|
|
435
|
+
await expect(async () => {
|
|
436
|
+
await act(async () => {
|
|
437
|
+
await useGraphStore.getState().generateAgentViewGraph();
|
|
438
|
+
});
|
|
439
|
+
}).rejects.toThrow('Backend API unavailable');
|
|
440
|
+
|
|
441
|
+
// Verify error state was set
|
|
442
|
+
const state = useGraphStore.getState();
|
|
443
|
+
expect(state.error).toBe('Backend API unavailable');
|
|
444
|
+
expect(state.isLoading).toBe(false);
|
|
445
|
+
});
|
|
446
|
+
|
|
447
|
+
it('should not crash when fetchGraphSnapshot returns invalid data', async () => {
|
|
448
|
+
// Mock returns empty snapshot
|
|
449
|
+
vi.mocked(graphService.fetchGraphSnapshot).mockResolvedValue({
|
|
450
|
+
generatedAt: '2025-10-11T00:00:00Z',
|
|
451
|
+
viewMode: 'agent',
|
|
452
|
+
filters: {
|
|
453
|
+
correlation_id: null,
|
|
454
|
+
time_range: { preset: 'last10min' },
|
|
455
|
+
artifactTypes: [],
|
|
456
|
+
producers: [],
|
|
457
|
+
tags: [],
|
|
458
|
+
visibility: [],
|
|
459
|
+
},
|
|
460
|
+
nodes: [],
|
|
461
|
+
edges: [],
|
|
462
|
+
statistics: null,
|
|
463
|
+
totalArtifacts: 0,
|
|
464
|
+
truncated: false,
|
|
465
|
+
});
|
|
466
|
+
|
|
467
|
+
await act(async () => {
|
|
468
|
+
await useGraphStore.getState().generateAgentViewGraph();
|
|
469
|
+
});
|
|
470
|
+
|
|
471
|
+
// Should handle empty data gracefully
|
|
472
|
+
const state = useGraphStore.getState();
|
|
473
|
+
expect(state.nodes).toEqual([]);
|
|
474
|
+
expect(state.edges).toEqual([]);
|
|
475
|
+
});
|
|
476
|
+
});
|
|
477
|
+
|
|
478
|
+
describe('Test 7: Empty State', () => {
|
|
479
|
+
it('should handle empty graph from backend', async () => {
|
|
480
|
+
const emptySnapshot = createMockSnapshot({
|
|
481
|
+
nodes: [],
|
|
482
|
+
edges: [],
|
|
483
|
+
statistics: null,
|
|
484
|
+
totalArtifacts: 0,
|
|
485
|
+
});
|
|
486
|
+
|
|
487
|
+
vi.mocked(graphService.fetchGraphSnapshot).mockResolvedValue(emptySnapshot);
|
|
488
|
+
|
|
489
|
+
await act(async () => {
|
|
490
|
+
await useGraphStore.getState().generateAgentViewGraph();
|
|
491
|
+
});
|
|
492
|
+
|
|
493
|
+
const state = useGraphStore.getState();
|
|
494
|
+
expect(state.nodes).toEqual([]);
|
|
495
|
+
expect(state.edges).toEqual([]);
|
|
496
|
+
expect(state.statistics).toBeNull();
|
|
497
|
+
});
|
|
498
|
+
|
|
499
|
+
it('should clear events when limit exceeded', () => {
|
|
500
|
+
useGraphStore.setState({ events: [] });
|
|
501
|
+
|
|
502
|
+
// Add 101 events (limit is 100)
|
|
503
|
+
act(() => {
|
|
504
|
+
for (let i = 0; i < 101; i++) {
|
|
505
|
+
useGraphStore.getState().addEvent({
|
|
506
|
+
id: `msg${i}`,
|
|
507
|
+
type: 'Pizza',
|
|
508
|
+
producedBy: 'pizza_master',
|
|
509
|
+
timestamp: Date.now() + i,
|
|
510
|
+
} as any);
|
|
511
|
+
}
|
|
512
|
+
});
|
|
513
|
+
|
|
514
|
+
// Should keep only last 100
|
|
515
|
+
const state = useGraphStore.getState();
|
|
516
|
+
expect(state.events).toHaveLength(100);
|
|
517
|
+
expect(state.events[0]?.id).toBe('msg100'); // Most recent first
|
|
518
|
+
});
|
|
519
|
+
});
|
|
520
|
+
|
|
521
|
+
describe('Test 8: View Mode Switching', () => {
|
|
522
|
+
it('should switch between agent and blackboard views', async () => {
|
|
523
|
+
const agentSnapshot = createMockSnapshot({ viewMode: 'agent' });
|
|
524
|
+
const blackboardSnapshot = createMockSnapshot({ viewMode: 'blackboard' });
|
|
525
|
+
|
|
526
|
+
vi.mocked(graphService.fetchGraphSnapshot)
|
|
527
|
+
.mockResolvedValueOnce(agentSnapshot)
|
|
528
|
+
.mockResolvedValueOnce(blackboardSnapshot);
|
|
529
|
+
|
|
530
|
+
// Load agent view
|
|
531
|
+
await act(async () => {
|
|
532
|
+
await useGraphStore.getState().generateAgentViewGraph();
|
|
533
|
+
});
|
|
534
|
+
|
|
535
|
+
expect(useGraphStore.getState().viewMode).toBe('agent');
|
|
536
|
+
|
|
537
|
+
// Switch to blackboard view
|
|
538
|
+
await act(async () => {
|
|
539
|
+
await useGraphStore.getState().generateBlackboardViewGraph();
|
|
540
|
+
});
|
|
541
|
+
|
|
542
|
+
expect(useGraphStore.getState().viewMode).toBe('blackboard');
|
|
543
|
+
expect(graphService.fetchGraphSnapshot).toHaveBeenCalledTimes(2);
|
|
544
|
+
});
|
|
545
|
+
|
|
546
|
+
it('should update viewMode state', () => {
|
|
547
|
+
act(() => {
|
|
548
|
+
useGraphStore.getState().setViewMode('blackboard');
|
|
549
|
+
});
|
|
550
|
+
|
|
551
|
+
expect(useGraphStore.getState().viewMode).toBe('blackboard');
|
|
552
|
+
|
|
553
|
+
act(() => {
|
|
554
|
+
useGraphStore.getState().setViewMode('agent');
|
|
555
|
+
});
|
|
556
|
+
|
|
557
|
+
expect(useGraphStore.getState().viewMode).toBe('agent');
|
|
558
|
+
});
|
|
559
|
+
});
|
|
560
|
+
|
|
561
|
+
describe('Test 9: Debounced Refresh (Critical Optimization)', () => {
|
|
562
|
+
beforeEach(() => {
|
|
563
|
+
vi.useFakeTimers();
|
|
564
|
+
});
|
|
565
|
+
|
|
566
|
+
afterEach(() => {
|
|
567
|
+
vi.useRealTimers();
|
|
568
|
+
});
|
|
569
|
+
|
|
570
|
+
it('should batch multiple rapid events into single backend fetch after 100ms', async () => {
|
|
571
|
+
const mockSnapshot = createMockSnapshot();
|
|
572
|
+
vi.mocked(graphService.fetchGraphSnapshot).mockResolvedValue(mockSnapshot);
|
|
573
|
+
|
|
574
|
+
// Load initial graph
|
|
575
|
+
await act(async () => {
|
|
576
|
+
await useGraphStore.getState().generateAgentViewGraph();
|
|
577
|
+
});
|
|
578
|
+
|
|
579
|
+
vi.clearAllMocks();
|
|
580
|
+
|
|
581
|
+
// Simulate 5 rapid scheduleRefresh() calls (like rapid WebSocket events)
|
|
582
|
+
act(() => {
|
|
583
|
+
for (let i = 0; i < 5; i++) {
|
|
584
|
+
useGraphStore.getState().scheduleRefresh();
|
|
585
|
+
}
|
|
586
|
+
});
|
|
587
|
+
|
|
588
|
+
// No fetch yet (debounce delay)
|
|
589
|
+
expect(graphService.fetchGraphSnapshot).not.toHaveBeenCalled();
|
|
590
|
+
|
|
591
|
+
// Advance timers by 100ms (debounce threshold)
|
|
592
|
+
await act(async () => {
|
|
593
|
+
vi.advanceTimersByTime(100);
|
|
594
|
+
// Wait for any pending promises
|
|
595
|
+
await Promise.resolve();
|
|
596
|
+
});
|
|
597
|
+
|
|
598
|
+
// Should have triggered exactly ONE backend fetch (batching worked!)
|
|
599
|
+
expect(graphService.fetchGraphSnapshot).toHaveBeenCalledTimes(1);
|
|
600
|
+
});
|
|
601
|
+
|
|
602
|
+
it('should reset debounce timer if new event arrives within 100ms', async () => {
|
|
603
|
+
const mockSnapshot = createMockSnapshot();
|
|
604
|
+
vi.mocked(graphService.fetchGraphSnapshot).mockResolvedValue(mockSnapshot);
|
|
605
|
+
|
|
606
|
+
await act(async () => {
|
|
607
|
+
await useGraphStore.getState().generateAgentViewGraph();
|
|
608
|
+
});
|
|
609
|
+
|
|
610
|
+
vi.clearAllMocks();
|
|
611
|
+
|
|
612
|
+
// First scheduleRefresh call
|
|
613
|
+
act(() => {
|
|
614
|
+
useGraphStore.getState().scheduleRefresh();
|
|
615
|
+
});
|
|
616
|
+
|
|
617
|
+
// Advance 50ms (not enough to trigger)
|
|
618
|
+
act(() => {
|
|
619
|
+
vi.advanceTimersByTime(50);
|
|
620
|
+
});
|
|
621
|
+
|
|
622
|
+
expect(graphService.fetchGraphSnapshot).not.toHaveBeenCalled();
|
|
623
|
+
|
|
624
|
+
// Second scheduleRefresh call (resets timer)
|
|
625
|
+
act(() => {
|
|
626
|
+
useGraphStore.getState().scheduleRefresh();
|
|
627
|
+
});
|
|
628
|
+
|
|
629
|
+
// Advance another 50ms (100ms total, but timer was reset at 50ms)
|
|
630
|
+
act(() => {
|
|
631
|
+
vi.advanceTimersByTime(50);
|
|
632
|
+
});
|
|
633
|
+
|
|
634
|
+
// Still no fetch (timer was reset)
|
|
635
|
+
expect(graphService.fetchGraphSnapshot).not.toHaveBeenCalled();
|
|
636
|
+
|
|
637
|
+
// Advance final 50ms (100ms since last scheduleRefresh)
|
|
638
|
+
await act(async () => {
|
|
639
|
+
vi.advanceTimersByTime(50);
|
|
640
|
+
await Promise.resolve();
|
|
641
|
+
});
|
|
642
|
+
|
|
643
|
+
// Now it should fetch (100ms of quiet time)
|
|
644
|
+
expect(graphService.fetchGraphSnapshot).toHaveBeenCalledTimes(1);
|
|
645
|
+
});
|
|
646
|
+
});
|
|
647
|
+
});
|
|
@@ -2,16 +2,18 @@ import React from 'react';
|
|
|
2
2
|
import { useUIStore } from '../../store/uiStore';
|
|
3
3
|
import { useGraphStore } from '../../store/graphStore';
|
|
4
4
|
import NodeDetailWindow from './NodeDetailWindow';
|
|
5
|
+
import MessageDetailWindow from './MessageDetailWindow';
|
|
5
6
|
|
|
6
7
|
/**
|
|
7
8
|
* Container component that renders all open detail windows.
|
|
8
9
|
* Manages multiple floating windows with independent drag/resize.
|
|
10
|
+
*
|
|
11
|
+
* Phase 6: Agent nodes show NodeDetailWindow (Live Output, Message History, Run Status tabs)
|
|
12
|
+
* Message nodes show MessageDetailWindow (Metadata, Payload, Consumption History)
|
|
9
13
|
*/
|
|
10
14
|
const DetailWindowContainer: React.FC = () => {
|
|
11
15
|
const detailWindows = useUIStore((state) => state.detailWindows);
|
|
12
|
-
const
|
|
13
|
-
const agents = useGraphStore((state) => state.agents);
|
|
14
|
-
const messages = useGraphStore((state) => state.messages);
|
|
16
|
+
const nodes = useGraphStore((state) => state.nodes);
|
|
15
17
|
|
|
16
18
|
// Convert Map to array for rendering
|
|
17
19
|
const windowEntries = Array.from(detailWindows.entries());
|
|
@@ -37,22 +39,16 @@ const DetailWindowContainer: React.FC = () => {
|
|
|
37
39
|
}}
|
|
38
40
|
>
|
|
39
41
|
{windowEntries.map(([nodeId, _window]) => {
|
|
40
|
-
//
|
|
41
|
-
|
|
42
|
+
// UI Optimization Migration (Phase 4.1): Read node type from state.nodes
|
|
43
|
+
const node = nodes.find((n) => n.id === nodeId);
|
|
44
|
+
const nodeType: 'agent' | 'message' = node?.type === 'agent' ? 'agent' : 'message';
|
|
42
45
|
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
}
|
|
48
|
-
} else if (mode === 'blackboard') {
|
|
49
|
-
// In blackboard view, nodes are messages
|
|
50
|
-
if (messages.has(nodeId)) {
|
|
51
|
-
nodeType = 'message';
|
|
52
|
-
}
|
|
46
|
+
// Render appropriate window based on node type
|
|
47
|
+
if (nodeType === 'message') {
|
|
48
|
+
return <MessageDetailWindow key={nodeId} nodeId={nodeId} />;
|
|
49
|
+
} else {
|
|
50
|
+
return <NodeDetailWindow key={nodeId} nodeId={nodeId} nodeType={nodeType} />;
|
|
53
51
|
}
|
|
54
|
-
|
|
55
|
-
return <NodeDetailWindow key={nodeId} nodeId={nodeId} nodeType={nodeType} />;
|
|
56
52
|
})}
|
|
57
53
|
</div>
|
|
58
54
|
</div>
|