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.

Files changed (117) hide show
  1. flock/dashboard/launcher.py +1 -1
  2. flock/frontend/README.md +678 -0
  3. flock/frontend/docs/DESIGN_SYSTEM.md +1980 -0
  4. flock/frontend/index.html +12 -0
  5. flock/frontend/package-lock.json +4347 -0
  6. flock/frontend/package.json +48 -0
  7. flock/frontend/src/App.tsx +79 -0
  8. flock/frontend/src/__tests__/e2e/critical-scenarios.test.tsx +587 -0
  9. flock/frontend/src/__tests__/integration/filtering-e2e.test.tsx +387 -0
  10. flock/frontend/src/__tests__/integration/graph-rendering.test.tsx +640 -0
  11. flock/frontend/src/__tests__/integration/indexeddb-persistence.test.tsx +699 -0
  12. flock/frontend/src/components/common/BuildInfo.tsx +39 -0
  13. flock/frontend/src/components/common/EmptyState.module.css +115 -0
  14. flock/frontend/src/components/common/EmptyState.tsx +128 -0
  15. flock/frontend/src/components/common/ErrorBoundary.module.css +169 -0
  16. flock/frontend/src/components/common/ErrorBoundary.tsx +118 -0
  17. flock/frontend/src/components/common/KeyboardShortcutsDialog.css +251 -0
  18. flock/frontend/src/components/common/KeyboardShortcutsDialog.tsx +151 -0
  19. flock/frontend/src/components/common/LoadingSpinner.module.css +97 -0
  20. flock/frontend/src/components/common/LoadingSpinner.tsx +29 -0
  21. flock/frontend/src/components/controls/PublishControl.css +547 -0
  22. flock/frontend/src/components/controls/PublishControl.test.tsx +543 -0
  23. flock/frontend/src/components/controls/PublishControl.tsx +432 -0
  24. flock/frontend/src/components/details/DetailWindowContainer.tsx +62 -0
  25. flock/frontend/src/components/details/LiveOutputTab.test.tsx +792 -0
  26. flock/frontend/src/components/details/LiveOutputTab.tsx +220 -0
  27. flock/frontend/src/components/details/MessageHistoryTab.tsx +299 -0
  28. flock/frontend/src/components/details/NodeDetailWindow.test.tsx +501 -0
  29. flock/frontend/src/components/details/NodeDetailWindow.tsx +218 -0
  30. flock/frontend/src/components/details/RunStatusTab.tsx +307 -0
  31. flock/frontend/src/components/details/tabs.test.tsx +1015 -0
  32. flock/frontend/src/components/filters/CorrelationIDFilter.module.css +102 -0
  33. flock/frontend/src/components/filters/CorrelationIDFilter.test.tsx +197 -0
  34. flock/frontend/src/components/filters/CorrelationIDFilter.tsx +121 -0
  35. flock/frontend/src/components/filters/FilterBar.module.css +29 -0
  36. flock/frontend/src/components/filters/FilterBar.test.tsx +133 -0
  37. flock/frontend/src/components/filters/FilterBar.tsx +33 -0
  38. flock/frontend/src/components/filters/FilterPills.module.css +79 -0
  39. flock/frontend/src/components/filters/FilterPills.test.tsx +173 -0
  40. flock/frontend/src/components/filters/FilterPills.tsx +67 -0
  41. flock/frontend/src/components/filters/TimeRangeFilter.module.css +91 -0
  42. flock/frontend/src/components/filters/TimeRangeFilter.test.tsx +154 -0
  43. flock/frontend/src/components/filters/TimeRangeFilter.tsx +105 -0
  44. flock/frontend/src/components/graph/AgentNode.test.tsx +75 -0
  45. flock/frontend/src/components/graph/AgentNode.tsx +322 -0
  46. flock/frontend/src/components/graph/GraphCanvas.tsx +406 -0
  47. flock/frontend/src/components/graph/MessageFlowEdge.tsx +128 -0
  48. flock/frontend/src/components/graph/MessageNode.test.tsx +62 -0
  49. flock/frontend/src/components/graph/MessageNode.tsx +116 -0
  50. flock/frontend/src/components/graph/MiniMap.tsx +47 -0
  51. flock/frontend/src/components/graph/TransformEdge.tsx +123 -0
  52. flock/frontend/src/components/layout/DashboardLayout.css +407 -0
  53. flock/frontend/src/components/layout/DashboardLayout.tsx +300 -0
  54. flock/frontend/src/components/layout/Header.module.css +88 -0
  55. flock/frontend/src/components/layout/Header.tsx +52 -0
  56. flock/frontend/src/components/modules/EventLogModule.test.tsx +401 -0
  57. flock/frontend/src/components/modules/EventLogModule.tsx +396 -0
  58. flock/frontend/src/components/modules/EventLogModuleWrapper.tsx +17 -0
  59. flock/frontend/src/components/modules/ModuleRegistry.test.ts +333 -0
  60. flock/frontend/src/components/modules/ModuleRegistry.ts +85 -0
  61. flock/frontend/src/components/modules/ModuleWindow.tsx +155 -0
  62. flock/frontend/src/components/modules/registerModules.ts +20 -0
  63. flock/frontend/src/components/settings/AdvancedSettings.tsx +175 -0
  64. flock/frontend/src/components/settings/AppearanceSettings.tsx +185 -0
  65. flock/frontend/src/components/settings/GraphSettings.tsx +110 -0
  66. flock/frontend/src/components/settings/SettingsPanel.css +327 -0
  67. flock/frontend/src/components/settings/SettingsPanel.tsx +131 -0
  68. flock/frontend/src/components/settings/ThemeSelector.tsx +298 -0
  69. flock/frontend/src/hooks/useKeyboardShortcuts.ts +148 -0
  70. flock/frontend/src/hooks/useModulePersistence.test.ts +442 -0
  71. flock/frontend/src/hooks/useModulePersistence.ts +154 -0
  72. flock/frontend/src/hooks/useModules.ts +139 -0
  73. flock/frontend/src/hooks/usePersistence.ts +139 -0
  74. flock/frontend/src/main.tsx +13 -0
  75. flock/frontend/src/services/api.ts +213 -0
  76. flock/frontend/src/services/indexeddb.test.ts +793 -0
  77. flock/frontend/src/services/indexeddb.ts +794 -0
  78. flock/frontend/src/services/layout.test.ts +437 -0
  79. flock/frontend/src/services/layout.ts +146 -0
  80. flock/frontend/src/services/themeApplicator.ts +140 -0
  81. flock/frontend/src/services/themeService.ts +77 -0
  82. flock/frontend/src/services/websocket.test.ts +595 -0
  83. flock/frontend/src/services/websocket.ts +685 -0
  84. flock/frontend/src/store/filterStore.test.ts +242 -0
  85. flock/frontend/src/store/filterStore.ts +103 -0
  86. flock/frontend/src/store/graphStore.test.ts +186 -0
  87. flock/frontend/src/store/graphStore.ts +414 -0
  88. flock/frontend/src/store/moduleStore.test.ts +253 -0
  89. flock/frontend/src/store/moduleStore.ts +57 -0
  90. flock/frontend/src/store/settingsStore.ts +188 -0
  91. flock/frontend/src/store/streamStore.ts +68 -0
  92. flock/frontend/src/store/uiStore.test.ts +54 -0
  93. flock/frontend/src/store/uiStore.ts +110 -0
  94. flock/frontend/src/store/wsStore.ts +34 -0
  95. flock/frontend/src/styles/index.css +15 -0
  96. flock/frontend/src/styles/scrollbar.css +47 -0
  97. flock/frontend/src/styles/variables.css +488 -0
  98. flock/frontend/src/test/setup.ts +1 -0
  99. flock/frontend/src/types/filters.ts +14 -0
  100. flock/frontend/src/types/graph.ts +55 -0
  101. flock/frontend/src/types/modules.ts +7 -0
  102. flock/frontend/src/types/theme.ts +55 -0
  103. flock/frontend/src/utils/mockData.ts +85 -0
  104. flock/frontend/src/utils/performance.ts +16 -0
  105. flock/frontend/src/utils/transforms.test.ts +860 -0
  106. flock/frontend/src/utils/transforms.ts +323 -0
  107. flock/frontend/src/vite-env.d.ts +17 -0
  108. flock/frontend/tsconfig.json +27 -0
  109. flock/frontend/tsconfig.node.json +11 -0
  110. flock/frontend/vite.config.ts +25 -0
  111. flock/frontend/vitest.config.ts +11 -0
  112. flock/helper/cli_helper.py +1 -1
  113. {flock_core-0.5.0b50.dist-info → flock_core-0.5.0b52.dist-info}/METADATA +1 -1
  114. {flock_core-0.5.0b50.dist-info → flock_core-0.5.0b52.dist-info}/RECORD +117 -7
  115. {flock_core-0.5.0b50.dist-info → flock_core-0.5.0b52.dist-info}/WHEEL +0 -0
  116. {flock_core-0.5.0b50.dist-info → flock_core-0.5.0b52.dist-info}/entry_points.txt +0 -0
  117. {flock_core-0.5.0b50.dist-info → flock_core-0.5.0b52.dist-info}/licenses/LICENSE +0 -0
@@ -0,0 +1,401 @@
1
+ import { describe, it, expect, vi, beforeEach } from 'vitest';
2
+ import { render, screen, within } from '@testing-library/react';
3
+ import userEvent from '@testing-library/user-event';
4
+ import EventLogModule from './EventLogModule';
5
+ import { Message, Agent } from '../../types/graph';
6
+ import { TimeRange } from '../../types/filters';
7
+
8
+ // Mock ModuleContext type based on the architecture spec
9
+ interface ModuleContext {
10
+ agents: Map<string, Agent>;
11
+ messages: Map<string, Message>;
12
+ events: Message[];
13
+ filters: {
14
+ correlationId: string | null;
15
+ timeRange: TimeRange;
16
+ };
17
+ publish: (artifact: any) => void;
18
+ invoke: (agentName: string, inputs: any[]) => void;
19
+ }
20
+
21
+ describe('EventLogModule', () => {
22
+ const createMockAgent = (id: string, name: string): Agent => ({
23
+ id,
24
+ name,
25
+ status: 'idle',
26
+ subscriptions: [],
27
+ lastActive: Date.now(),
28
+ sentCount: 0,
29
+ recvCount: 0,
30
+ });
31
+
32
+ const createMockMessage = (
33
+ id: string,
34
+ type: string,
35
+ producedBy: string,
36
+ correlationId: string,
37
+ timestamp: number
38
+ ): Message => ({
39
+ id,
40
+ type,
41
+ payload: { data: `test-${id}` },
42
+ timestamp,
43
+ correlationId,
44
+ producedBy,
45
+ });
46
+
47
+ const createMockContext = (
48
+ events: Message[] = [],
49
+ correlationId: string | null = null,
50
+ timeRange: TimeRange = { preset: 'last10min' }
51
+ ): ModuleContext => ({
52
+ agents: new Map([
53
+ ['agent1', createMockAgent('agent1', 'TestAgent1')],
54
+ ['agent2', createMockAgent('agent2', 'TestAgent2')],
55
+ ]),
56
+ messages: new Map(),
57
+ events,
58
+ filters: {
59
+ correlationId,
60
+ timeRange,
61
+ },
62
+ publish: vi.fn(),
63
+ invoke: vi.fn(),
64
+ });
65
+
66
+ beforeEach(() => {
67
+ vi.clearAllMocks();
68
+ });
69
+
70
+ it('should render table with correct column headers', () => {
71
+ const context = createMockContext();
72
+ render(<EventLogModule context={context} />);
73
+
74
+ expect(screen.getByText(/Timestamp/)).toBeInTheDocument();
75
+ expect(screen.getByText(/^Type/)).toBeInTheDocument();
76
+ expect(screen.getByText(/^Agent/)).toBeInTheDocument();
77
+ expect(screen.getByText(/Correlation ID/)).toBeInTheDocument();
78
+ });
79
+
80
+ it('should display events from context in table rows', () => {
81
+ const now = Date.now();
82
+ const events = [
83
+ createMockMessage('msg1', 'Movie', 'agent1', 'corr-123', now - 1000),
84
+ createMockMessage('msg2', 'Tagline', 'agent2', 'corr-123', now),
85
+ ];
86
+ const context = createMockContext(events);
87
+
88
+ render(<EventLogModule context={context} />);
89
+
90
+ expect(screen.getByText('Movie')).toBeInTheDocument();
91
+ expect(screen.getByText('Tagline')).toBeInTheDocument();
92
+ });
93
+
94
+ it('should display timestamp for each event', () => {
95
+ const now = Date.now();
96
+ const events = [createMockMessage('msg1', 'Movie', 'agent1', 'corr-123', now - 1000)];
97
+ const context = createMockContext(events);
98
+
99
+ render(<EventLogModule context={context} />);
100
+
101
+ // Should display formatted timestamp - check it's in the table
102
+ const cells = screen.getAllByRole('cell');
103
+ // First cell should be the timestamp - just verify it exists and has some content
104
+ expect(cells[0]).toBeInTheDocument();
105
+ expect(cells[0]!.textContent).toBeTruthy();
106
+ });
107
+
108
+ it('should display event type for each event', () => {
109
+ const events = [
110
+ createMockMessage('msg1', 'Movie', 'agent1', 'corr-123', Date.now()),
111
+ createMockMessage('msg2', 'Tagline', 'agent2', 'corr-456', Date.now()),
112
+ ];
113
+ const context = createMockContext(events);
114
+
115
+ render(<EventLogModule context={context} />);
116
+
117
+ expect(screen.getByText('Movie')).toBeInTheDocument();
118
+ expect(screen.getByText('Tagline')).toBeInTheDocument();
119
+ });
120
+
121
+ it('should display agent name for each event', () => {
122
+ const events = [
123
+ createMockMessage('msg1', 'Movie', 'agent1', 'corr-123', Date.now()),
124
+ createMockMessage('msg2', 'Tagline', 'agent2', 'corr-123', Date.now()),
125
+ ];
126
+ const context = createMockContext(events);
127
+
128
+ render(<EventLogModule context={context} />);
129
+
130
+ expect(screen.getByText('agent1')).toBeInTheDocument();
131
+ expect(screen.getByText('agent2')).toBeInTheDocument();
132
+ });
133
+
134
+ it('should display correlation ID for each event', () => {
135
+ const events = [
136
+ createMockMessage('msg1', 'Movie', 'agent1', 'corr-123', Date.now()),
137
+ createMockMessage('msg2', 'Tagline', 'agent2', 'corr-456', Date.now()),
138
+ ];
139
+ const context = createMockContext(events);
140
+
141
+ render(<EventLogModule context={context} />);
142
+
143
+ expect(screen.getByText('corr-123')).toBeInTheDocument();
144
+ expect(screen.getByText('corr-456')).toBeInTheDocument();
145
+ });
146
+
147
+ it('should filter events by correlation ID from context', () => {
148
+ const events = [
149
+ createMockMessage('msg1', 'Movie', 'agent1', 'corr-123', Date.now()),
150
+ createMockMessage('msg2', 'Tagline', 'agent2', 'corr-456', Date.now()),
151
+ createMockMessage('msg3', 'Idea', 'agent1', 'corr-123', Date.now()),
152
+ ];
153
+ const context = createMockContext(events, 'corr-123');
154
+
155
+ render(<EventLogModule context={context} />);
156
+
157
+ // Should show events with corr-123
158
+ expect(screen.getByText('Movie')).toBeInTheDocument();
159
+ expect(screen.getByText('Idea')).toBeInTheDocument();
160
+
161
+ // Should NOT show event with corr-456
162
+ expect(screen.queryByText('Tagline')).not.toBeInTheDocument();
163
+ });
164
+
165
+ it('should filter events by time range from context', () => {
166
+ const now = Date.now();
167
+ const fifteenMinutesAgo = now - 15 * 60 * 1000;
168
+
169
+ const events = [
170
+ createMockMessage('msg1', 'Recent', 'agent1', 'corr-123', now - 1000),
171
+ createMockMessage('msg2', 'Old', 'agent2', 'corr-123', fifteenMinutesAgo),
172
+ ];
173
+
174
+ // Filter: last 10 minutes
175
+ const timeRange: TimeRange = { preset: 'last10min' };
176
+ const context = createMockContext(events, null, timeRange);
177
+
178
+ render(<EventLogModule context={context} />);
179
+
180
+ // Should show recent event
181
+ expect(screen.getByText('Recent')).toBeInTheDocument();
182
+
183
+ // Should NOT show old event (15 minutes ago)
184
+ expect(screen.queryByText('Old')).not.toBeInTheDocument();
185
+ });
186
+
187
+ it('should filter events by custom time range', () => {
188
+ const now = Date.now();
189
+ const events = [
190
+ createMockMessage('msg1', 'InRange', 'agent1', 'corr-123', now - 2000),
191
+ createMockMessage('msg2', 'OutOfRange', 'agent2', 'corr-123', now - 10000),
192
+ ];
193
+
194
+ // Custom time range: last 5 seconds
195
+ const timeRange: TimeRange = {
196
+ preset: 'custom',
197
+ start: now - 5000,
198
+ end: now,
199
+ };
200
+ const context = createMockContext(events, null, timeRange);
201
+
202
+ render(<EventLogModule context={context} />);
203
+
204
+ // Should show in-range event
205
+ expect(screen.getByText('InRange')).toBeInTheDocument();
206
+
207
+ // Should NOT show out-of-range event
208
+ expect(screen.queryByText('OutOfRange')).not.toBeInTheDocument();
209
+ });
210
+
211
+ it('should have expand button for each event row', () => {
212
+ const events = [
213
+ createMockMessage('msg1', 'Movie', 'agent1', 'corr-123', Date.now()),
214
+ createMockMessage('msg2', 'Tagline', 'agent2', 'corr-123', Date.now()),
215
+ ];
216
+ const context = createMockContext(events);
217
+
218
+ render(<EventLogModule context={context} />);
219
+
220
+ // Should have 2 row expand buttons + 1 "Expand All" button = 3 total
221
+ const allExpandButtons = screen.getAllByRole('button', { name: /expand/i });
222
+ expect(allExpandButtons).toHaveLength(3);
223
+
224
+ // Verify row expand buttons exist (there should be 2)
225
+ const rowExpandButtons = screen.getAllByRole('button', { name: /expand ▼/i });
226
+ expect(rowExpandButtons).toHaveLength(2);
227
+ });
228
+
229
+ it('should show empty state when no events', () => {
230
+ const context = createMockContext([]);
231
+
232
+ render(<EventLogModule context={context} />);
233
+
234
+ // Table headers should still be present
235
+ expect(screen.getByText(/Timestamp/)).toBeInTheDocument();
236
+
237
+ // Should show empty state message
238
+ expect(screen.getByText('No events to display')).toBeInTheDocument();
239
+ });
240
+
241
+ it('should show empty state when all events are filtered out', () => {
242
+ const events = [
243
+ createMockMessage('msg1', 'Movie', 'agent1', 'corr-123', Date.now()),
244
+ createMockMessage('msg2', 'Tagline', 'agent2', 'corr-456', Date.now()),
245
+ ];
246
+ // Filter by non-existent correlation ID
247
+ const context = createMockContext(events, 'corr-999');
248
+
249
+ render(<EventLogModule context={context} />);
250
+
251
+ // Should not show any events
252
+ expect(screen.queryByText('Movie')).not.toBeInTheDocument();
253
+ expect(screen.queryByText('Tagline')).not.toBeInTheDocument();
254
+ });
255
+
256
+ it('should have sortable column headers', () => {
257
+ const context = createMockContext();
258
+ render(<EventLogModule context={context} />);
259
+
260
+ const timestampHeader = screen.getByText(/Timestamp/);
261
+ const typeHeader = screen.getByText(/^Type/);
262
+ const agentHeader = screen.getByText(/^Agent/);
263
+
264
+ // Check if headers are clickable (have onClick or role)
265
+ expect(timestampHeader.closest('th')).toBeInTheDocument();
266
+ expect(typeHeader.closest('th')).toBeInTheDocument();
267
+ expect(agentHeader.closest('th')).toBeInTheDocument();
268
+ });
269
+
270
+ it('should sort events by timestamp when timestamp column is clicked', async () => {
271
+ const user = userEvent.setup();
272
+ const now = Date.now();
273
+ const events = [
274
+ createMockMessage('msg1', 'Movie', 'agent1', 'corr-123', now - 5000),
275
+ createMockMessage('msg2', 'Tagline', 'agent2', 'corr-123', now - 1000),
276
+ createMockMessage('msg3', 'Idea', 'agent1', 'corr-123', now - 3000),
277
+ ];
278
+ const context = createMockContext(events);
279
+
280
+ render(<EventLogModule context={context} />);
281
+
282
+ const timestampHeader = screen.getByText(/Timestamp/);
283
+
284
+ // Click to sort ascending
285
+ await user.click(timestampHeader);
286
+
287
+ const rows = screen.getAllByRole('row');
288
+ // First row is header, so data rows start at index 1
289
+ // Oldest should be first after ascending sort
290
+ expect(within(rows[1]!).getByText('Movie')).toBeInTheDocument();
291
+ expect(within(rows[2]!).getByText('Idea')).toBeInTheDocument();
292
+ expect(within(rows[3]!).getByText('Tagline')).toBeInTheDocument();
293
+ });
294
+
295
+ it('should sort events by type when type column is clicked', async () => {
296
+ const user = userEvent.setup();
297
+ const now = Date.now();
298
+ const events = [
299
+ createMockMessage('msg1', 'Zebra', 'agent1', 'corr-123', now),
300
+ createMockMessage('msg2', 'Apple', 'agent2', 'corr-123', now),
301
+ createMockMessage('msg3', 'Movie', 'agent1', 'corr-123', now),
302
+ ];
303
+ const context = createMockContext(events);
304
+
305
+ render(<EventLogModule context={context} />);
306
+
307
+ const typeHeader = screen.getByText(/^Type/);
308
+
309
+ // Click to sort ascending
310
+ await user.click(typeHeader);
311
+
312
+ const rows = screen.getAllByRole('row');
313
+ // After alphabetical sort: Apple, Movie, Zebra
314
+ expect(within(rows[1]!).getByText('Apple')).toBeInTheDocument();
315
+ expect(within(rows[2]!).getByText('Movie')).toBeInTheDocument();
316
+ expect(within(rows[3]!).getByText('Zebra')).toBeInTheDocument();
317
+ });
318
+
319
+ it('should sort events by agent name when agent column is clicked', async () => {
320
+ const user = userEvent.setup();
321
+ const now = Date.now();
322
+ const events = [
323
+ createMockMessage('msg1', 'Movie', 'zebra-agent', 'corr-123', now),
324
+ createMockMessage('msg2', 'Tagline', 'apple-agent', 'corr-123', now),
325
+ createMockMessage('msg3', 'Idea', 'movie-agent', 'corr-123', now),
326
+ ];
327
+ const context = createMockContext(events);
328
+
329
+ render(<EventLogModule context={context} />);
330
+
331
+ const agentHeader = screen.getByText(/^Agent/);
332
+
333
+ // Click to sort ascending
334
+ await user.click(agentHeader);
335
+
336
+ const rows = screen.getAllByRole('row');
337
+ // After alphabetical sort: apple-agent, movie-agent, zebra-agent
338
+ expect(within(rows[1]!).getByText('apple-agent')).toBeInTheDocument();
339
+ expect(within(rows[2]!).getByText('movie-agent')).toBeInTheDocument();
340
+ expect(within(rows[3]!).getByText('zebra-agent')).toBeInTheDocument();
341
+ });
342
+
343
+ it('should expand row to show event payload when expand button is clicked', async () => {
344
+ const user = userEvent.setup();
345
+ const events = [
346
+ createMockMessage('msg1', 'Movie', 'agent1', 'corr-123', Date.now()),
347
+ ];
348
+ const context = createMockContext(events);
349
+
350
+ render(<EventLogModule context={context} />);
351
+
352
+ // Get the row expand button (there's only one row, so one button)
353
+ const rowExpandButtons = screen.getAllByRole('button', { name: /expand ▼/i });
354
+ expect(rowExpandButtons.length).toBeGreaterThan(0);
355
+ const expandButton = rowExpandButtons[0]!; // Non-null assertion since we verified length
356
+
357
+ // Click expand button
358
+ await user.click(expandButton);
359
+
360
+ // Should show payload data
361
+ expect(screen.getByText(/test-msg1/i)).toBeInTheDocument();
362
+ });
363
+
364
+ it('should display events in reverse chronological order by default', () => {
365
+ const now = Date.now();
366
+ const events = [
367
+ createMockMessage('msg1', 'Oldest', 'agent1', 'corr-123', now - 10000),
368
+ createMockMessage('msg2', 'Middle', 'agent2', 'corr-123', now - 5000),
369
+ createMockMessage('msg3', 'Newest', 'agent1', 'corr-123', now),
370
+ ];
371
+ const context = createMockContext(events);
372
+
373
+ render(<EventLogModule context={context} />);
374
+
375
+ const rows = screen.getAllByRole('row');
376
+ // First row is header, so data rows start at index 1
377
+ // Newest should be first (reverse chronological)
378
+ expect(within(rows[1]!).getByText('Newest')).toBeInTheDocument();
379
+ expect(within(rows[2]!).getByText('Middle')).toBeInTheDocument();
380
+ expect(within(rows[3]!).getByText('Oldest')).toBeInTheDocument();
381
+ });
382
+
383
+ it('should apply both correlation ID and time range filters together', () => {
384
+ const now = Date.now();
385
+ const events = [
386
+ createMockMessage('msg1', 'Match', 'agent1', 'corr-123', now - 1000),
387
+ createMockMessage('msg2', 'WrongCorr', 'agent2', 'corr-456', now - 1000),
388
+ createMockMessage('msg3', 'WrongTime', 'agent1', 'corr-123', now - 20 * 60 * 1000),
389
+ ];
390
+
391
+ const timeRange: TimeRange = { preset: 'last10min' };
392
+ const context = createMockContext(events, 'corr-123', timeRange);
393
+
394
+ render(<EventLogModule context={context} />);
395
+
396
+ // Should only show event matching both filters
397
+ expect(screen.getByText('Match')).toBeInTheDocument();
398
+ expect(screen.queryByText('WrongCorr')).not.toBeInTheDocument();
399
+ expect(screen.queryByText('WrongTime')).not.toBeInTheDocument();
400
+ });
401
+ });