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,218 @@
1
+ import React, { useCallback } from 'react';
2
+ import { Rnd } from 'react-rnd';
3
+ import { useUIStore } from '../../store/uiStore';
4
+ import LiveOutputTab from './LiveOutputTab';
5
+ import MessageHistoryTab from './MessageHistoryTab';
6
+ import RunStatusTab from './RunStatusTab';
7
+
8
+ interface NodeDetailWindowProps {
9
+ nodeId: string;
10
+ nodeType?: 'agent' | 'message';
11
+ }
12
+
13
+ const NodeDetailWindow: React.FC<NodeDetailWindowProps> = ({ nodeId, nodeType = 'agent' }) => {
14
+ const window = useUIStore((state) => state.detailWindows.get(nodeId));
15
+ const updateDetailWindow = useUIStore((state) => state.updateDetailWindow);
16
+ const closeDetailWindow = useUIStore((state) => state.closeDetailWindow);
17
+
18
+ const handleClose = useCallback(() => {
19
+ closeDetailWindow(nodeId);
20
+ }, [nodeId, closeDetailWindow]);
21
+
22
+ const handleTabSwitch = useCallback(
23
+ (tab: 'liveOutput' | 'messageHistory' | 'runStatus') => {
24
+ updateDetailWindow(nodeId, { activeTab: tab });
25
+ },
26
+ [nodeId, updateDetailWindow]
27
+ );
28
+
29
+ if (!window) return null;
30
+
31
+ const { position, size, activeTab } = window;
32
+
33
+ return (
34
+ <Rnd
35
+ position={position}
36
+ size={size}
37
+ onDragStop={(_e, d) => {
38
+ updateDetailWindow(nodeId, {
39
+ position: { x: d.x, y: d.y },
40
+ });
41
+ }}
42
+ onResizeStop={(_e, _direction, ref, _delta, position) => {
43
+ updateDetailWindow(nodeId, {
44
+ size: {
45
+ width: parseInt(ref.style.width, 10),
46
+ height: parseInt(ref.style.height, 10),
47
+ },
48
+ position,
49
+ });
50
+ }}
51
+ minWidth={600}
52
+ minHeight={400}
53
+ bounds="parent"
54
+ dragHandleClassName="window-header"
55
+ style={{
56
+ zIndex: 1000,
57
+ display: 'flex',
58
+ flexDirection: 'column',
59
+ pointerEvents: 'all',
60
+ }}
61
+ >
62
+ <div
63
+ style={{
64
+ display: 'flex',
65
+ flexDirection: 'column',
66
+ width: '100%',
67
+ height: '100%',
68
+ background: 'var(--color-glass-bg)',
69
+ border: 'var(--border-width-1) solid var(--color-glass-border)',
70
+ borderRadius: 'var(--radius-xl)',
71
+ overflow: 'hidden',
72
+ boxShadow: 'var(--shadow-xl)',
73
+ backdropFilter: 'blur(var(--blur-lg))',
74
+ WebkitBackdropFilter: 'blur(var(--blur-lg))',
75
+ }}
76
+ >
77
+ {/* Header */}
78
+ <div
79
+ className="window-header"
80
+ style={{
81
+ display: 'flex',
82
+ alignItems: 'center',
83
+ justifyContent: 'space-between',
84
+ padding: 'var(--space-component-md) var(--space-component-lg)',
85
+ background: 'rgba(42, 42, 50, 0.5)',
86
+ borderBottom: 'var(--border-width-1) solid var(--color-border-subtle)',
87
+ cursor: 'move',
88
+ userSelect: 'none',
89
+ }}
90
+ >
91
+ <div style={{ display: 'flex', alignItems: 'center', gap: 'var(--gap-md)' }}>
92
+ <div
93
+ style={{
94
+ width: '10px',
95
+ height: '10px',
96
+ borderRadius: 'var(--radius-circle)',
97
+ background: nodeType === 'agent' ? 'var(--color-primary-500)' : 'var(--color-warning)',
98
+ boxShadow: nodeType === 'agent'
99
+ ? '0 0 8px var(--color-primary-500)'
100
+ : '0 0 8px var(--color-warning)',
101
+ }}
102
+ />
103
+ <span
104
+ style={{
105
+ color: 'var(--color-text-primary)',
106
+ fontSize: 'var(--font-size-body-sm)',
107
+ fontWeight: 'var(--font-weight-semibold)',
108
+ fontFamily: 'var(--font-family-sans)',
109
+ }}
110
+ >
111
+ {nodeType === 'agent' ? 'Agent' : 'Message'}: {nodeId}
112
+ </span>
113
+ </div>
114
+ <button
115
+ onClick={handleClose}
116
+ aria-label="Close window"
117
+ style={{
118
+ background: 'transparent',
119
+ border: 'none',
120
+ color: 'var(--color-text-secondary)',
121
+ fontSize: 'var(--font-size-h3)',
122
+ cursor: 'pointer',
123
+ padding: 'var(--spacing-1) var(--spacing-2)',
124
+ lineHeight: 1,
125
+ borderRadius: 'var(--radius-md)',
126
+ transition: 'var(--transition-colors)',
127
+ display: 'flex',
128
+ alignItems: 'center',
129
+ justifyContent: 'center',
130
+ }}
131
+ onMouseEnter={(e) => {
132
+ e.currentTarget.style.color = 'var(--color-error)';
133
+ e.currentTarget.style.background = 'var(--color-error-bg)';
134
+ }}
135
+ onMouseLeave={(e) => {
136
+ e.currentTarget.style.color = 'var(--color-text-secondary)';
137
+ e.currentTarget.style.background = 'transparent';
138
+ }}
139
+ >
140
+ ×
141
+ </button>
142
+ </div>
143
+
144
+ {/* Tabs */}
145
+ <div
146
+ style={{
147
+ display: 'flex',
148
+ gap: 'var(--gap-sm)',
149
+ padding: 'var(--space-component-sm) var(--space-component-lg)',
150
+ background: 'rgba(35, 35, 41, 0.4)',
151
+ borderBottom: 'var(--border-width-1) solid var(--color-border-subtle)',
152
+ }}
153
+ >
154
+ {(['liveOutput', 'messageHistory', 'runStatus'] as const).map((tab) => {
155
+ const isActive = activeTab === tab;
156
+ return (
157
+ <button
158
+ key={tab}
159
+ onClick={() => handleTabSwitch(tab)}
160
+ style={{
161
+ position: 'relative',
162
+ padding: 'var(--space-component-sm) var(--space-component-md)',
163
+ background: isActive ? 'var(--color-bg-surface)' : 'transparent',
164
+ border: 'none',
165
+ borderRadius: 'var(--radius-md)',
166
+ color: isActive ? 'var(--color-text-primary)' : 'var(--color-text-secondary)',
167
+ fontSize: 'var(--font-size-body-sm)',
168
+ fontWeight: isActive ? 'var(--font-weight-semibold)' : 'var(--font-weight-medium)',
169
+ fontFamily: 'var(--font-family-sans)',
170
+ cursor: 'pointer',
171
+ transition: 'var(--transition-colors)',
172
+ borderBottom: isActive
173
+ ? '2px solid var(--color-primary-500)'
174
+ : '2px solid transparent',
175
+ }}
176
+ onMouseEnter={(e) => {
177
+ if (!isActive) {
178
+ e.currentTarget.style.background = 'var(--color-bg-elevated)';
179
+ e.currentTarget.style.color = 'var(--color-text-primary)';
180
+ }
181
+ }}
182
+ onMouseLeave={(e) => {
183
+ if (!isActive) {
184
+ e.currentTarget.style.background = 'transparent';
185
+ e.currentTarget.style.color = 'var(--color-text-secondary)';
186
+ }
187
+ }}
188
+ >
189
+ {tab === 'liveOutput'
190
+ ? 'Live Output'
191
+ : tab === 'messageHistory'
192
+ ? 'Message History'
193
+ : 'Run Status'}
194
+ </button>
195
+ );
196
+ })}
197
+ </div>
198
+
199
+ {/* Tab Content */}
200
+ <div
201
+ style={{
202
+ flex: 1,
203
+ overflow: 'hidden',
204
+ background: 'var(--color-bg-surface)',
205
+ }}
206
+ >
207
+ {activeTab === 'liveOutput' && <LiveOutputTab nodeId={nodeId} nodeType={nodeType} />}
208
+ {activeTab === 'messageHistory' && (
209
+ <MessageHistoryTab nodeId={nodeId} nodeType={nodeType} />
210
+ )}
211
+ {activeTab === 'runStatus' && <RunStatusTab nodeId={nodeId} nodeType={nodeType} />}
212
+ </div>
213
+ </div>
214
+ </Rnd>
215
+ );
216
+ };
217
+
218
+ export default NodeDetailWindow;
@@ -0,0 +1,307 @@
1
+ import React, { useMemo } from 'react';
2
+ import { useGraphStore } from '../../store/graphStore';
3
+
4
+ interface RunStatusTabProps {
5
+ nodeId: string;
6
+ nodeType: 'agent' | 'message';
7
+ }
8
+
9
+ interface RunStatusEntry {
10
+ runId: string;
11
+ startTime: number;
12
+ endTime: number;
13
+ duration: number;
14
+ status: 'idle' | 'processing' | 'error';
15
+ metrics: {
16
+ tokensUsed?: number;
17
+ costUsd?: number;
18
+ artifactsProduced?: number;
19
+ };
20
+ errorMessage?: string;
21
+ }
22
+
23
+ const RunStatusTab: React.FC<RunStatusTabProps> = ({ nodeId, nodeType }) => {
24
+ const runs = useGraphStore((state) => state.runs);
25
+
26
+ // Build run history for this agent
27
+ const runHistory = useMemo(() => {
28
+ const history: RunStatusEntry[] = [];
29
+
30
+ if (nodeType !== 'agent') {
31
+ return history;
32
+ }
33
+
34
+ runs.forEach((run) => {
35
+ // Filter runs for this agent
36
+ if (run.agent_name === nodeId) {
37
+ const startTime = run.started_at ? new Date(run.started_at).getTime() : Date.now();
38
+ const endTime = run.completed_at ? new Date(run.completed_at).getTime() : Date.now();
39
+
40
+ // Map status
41
+ let status: 'idle' | 'processing' | 'error' = 'idle';
42
+ if (run.status === 'active') {
43
+ status = 'processing';
44
+ } else if (run.status === 'error') {
45
+ status = 'error';
46
+ } else {
47
+ status = 'idle';
48
+ }
49
+
50
+ history.push({
51
+ runId: run.run_id,
52
+ startTime,
53
+ endTime,
54
+ duration: run.duration_ms || 0,
55
+ status,
56
+ metrics: run.metrics || {},
57
+ errorMessage: run.error_message || undefined,
58
+ });
59
+ }
60
+ });
61
+
62
+ // Sort by start time (most recent first)
63
+ return history.sort((a, b) => b.startTime - a.startTime);
64
+ }, [nodeId, nodeType, runs]);
65
+
66
+ const formatTimestamp = (timestamp: number) => {
67
+ return new Date(timestamp).toLocaleString();
68
+ };
69
+
70
+ const formatDuration = (ms: number) => {
71
+ if (ms < 1000) {
72
+ return `${ms}ms`;
73
+ } else if (ms < 60000) {
74
+ return `${(ms / 1000).toFixed(2)}s`;
75
+ } else {
76
+ return `${(ms / 60000).toFixed(2)}m`;
77
+ }
78
+ };
79
+
80
+ const getStatusColor = (status: RunStatusEntry['status']) => {
81
+ switch (status) {
82
+ case 'idle':
83
+ return 'var(--color-success-light)';
84
+ case 'processing':
85
+ return 'var(--color-tertiary-400)';
86
+ case 'error':
87
+ return 'var(--color-error-light)';
88
+ default:
89
+ return 'var(--color-text-primary)';
90
+ }
91
+ };
92
+
93
+ const getStatusLabel = (status: RunStatusEntry['status']) => {
94
+ switch (status) {
95
+ case 'idle':
96
+ return 'Completed';
97
+ case 'processing':
98
+ return 'Processing';
99
+ case 'error':
100
+ return 'Error';
101
+ default:
102
+ return status;
103
+ }
104
+ };
105
+
106
+ return (
107
+ <div
108
+ data-testid={`run-status-${nodeId}`}
109
+ style={{
110
+ height: '100%',
111
+ overflow: 'auto',
112
+ background: 'var(--color-bg-elevated)',
113
+ color: 'var(--color-text-primary)',
114
+ }}
115
+ >
116
+ {runHistory.length === 0 ? (
117
+ <div
118
+ data-testid="empty-runs"
119
+ style={{
120
+ padding: 'var(--space-layout-md)',
121
+ color: 'var(--color-text-muted)',
122
+ fontSize: 'var(--font-size-body-sm)',
123
+ fontFamily: 'var(--font-family-sans)',
124
+ textAlign: 'center',
125
+ }}
126
+ >
127
+ No previous runs
128
+ </div>
129
+ ) : (
130
+ <table
131
+ data-testid="run-table"
132
+ style={{
133
+ width: '100%',
134
+ borderCollapse: 'collapse',
135
+ fontSize: 'var(--font-size-caption)',
136
+ fontFamily: 'var(--font-family-sans)',
137
+ }}
138
+ >
139
+ <thead>
140
+ <tr
141
+ style={{
142
+ background: 'var(--color-bg-surface)',
143
+ borderBottom: 'var(--border-width-1) solid var(--color-border-subtle)',
144
+ position: 'sticky',
145
+ top: 0,
146
+ zIndex: 1,
147
+ }}
148
+ >
149
+ <th
150
+ style={{
151
+ padding: 'var(--space-component-sm) var(--space-component-md)',
152
+ textAlign: 'left',
153
+ fontWeight: 'var(--font-weight-semibold)',
154
+ color: 'var(--color-text-secondary)',
155
+ }}
156
+ >
157
+ Run ID
158
+ </th>
159
+ <th
160
+ style={{
161
+ padding: 'var(--space-component-sm) var(--space-component-md)',
162
+ textAlign: 'left',
163
+ fontWeight: 'var(--font-weight-semibold)',
164
+ color: 'var(--color-text-secondary)',
165
+ }}
166
+ >
167
+ Start Time
168
+ </th>
169
+ <th
170
+ style={{
171
+ padding: 'var(--space-component-sm) var(--space-component-md)',
172
+ textAlign: 'left',
173
+ fontWeight: 'var(--font-weight-semibold)',
174
+ color: 'var(--color-text-secondary)',
175
+ }}
176
+ >
177
+ Duration
178
+ </th>
179
+ <th
180
+ style={{
181
+ padding: 'var(--space-component-sm) var(--space-component-md)',
182
+ textAlign: 'left',
183
+ fontWeight: 'var(--font-weight-semibold)',
184
+ color: 'var(--color-text-secondary)',
185
+ }}
186
+ >
187
+ Status
188
+ </th>
189
+ <th
190
+ style={{
191
+ padding: 'var(--space-component-sm) var(--space-component-md)',
192
+ textAlign: 'left',
193
+ fontWeight: 'var(--font-weight-semibold)',
194
+ color: 'var(--color-text-secondary)',
195
+ }}
196
+ >
197
+ Metrics
198
+ </th>
199
+ </tr>
200
+ </thead>
201
+ <tbody>
202
+ {runHistory.map((run) => (
203
+ <tr
204
+ key={run.runId}
205
+ data-testid={`run-row-${run.runId}`}
206
+ style={{
207
+ borderBottom: 'var(--border-width-1) solid var(--color-border-subtle)',
208
+ transition: 'var(--transition-colors)',
209
+ }}
210
+ onMouseEnter={(e) => {
211
+ e.currentTarget.style.background = 'var(--color-bg-surface)';
212
+ }}
213
+ onMouseLeave={(e) => {
214
+ e.currentTarget.style.background = 'transparent';
215
+ }}
216
+ >
217
+ <td
218
+ data-testid={`run-id-${run.runId}`}
219
+ style={{
220
+ padding: 'var(--space-component-sm) var(--space-component-md)',
221
+ fontFamily: 'var(--font-family-mono)',
222
+ fontSize: 'var(--font-size-overline)',
223
+ maxWidth: '150px',
224
+ overflow: 'hidden',
225
+ textOverflow: 'ellipsis',
226
+ color: 'var(--color-text-muted)',
227
+ }}
228
+ >
229
+ {run.runId}
230
+ </td>
231
+ <td
232
+ data-testid={`run-start-${run.runId}`}
233
+ style={{
234
+ padding: 'var(--space-component-sm) var(--space-component-md)',
235
+ whiteSpace: 'nowrap',
236
+ color: 'var(--color-text-tertiary)',
237
+ }}
238
+ >
239
+ {formatTimestamp(run.startTime)}
240
+ </td>
241
+ <td
242
+ data-testid={`run-duration-${run.runId}`}
243
+ style={{
244
+ padding: 'var(--space-component-sm) var(--space-component-md)',
245
+ fontFamily: 'var(--font-family-mono)',
246
+ color: 'var(--color-text-secondary)',
247
+ }}
248
+ >
249
+ {formatDuration(run.duration)}
250
+ </td>
251
+ <td
252
+ data-testid={`run-status-${run.runId}`}
253
+ style={{
254
+ padding: 'var(--space-component-sm) var(--space-component-md)',
255
+ color: getStatusColor(run.status),
256
+ fontWeight: 'var(--font-weight-semibold)',
257
+ }}
258
+ >
259
+ {getStatusLabel(run.status)}
260
+ {run.errorMessage && (
261
+ <div
262
+ style={{
263
+ fontSize: 'var(--font-size-overline)',
264
+ color: 'var(--color-text-muted)',
265
+ marginTop: 'var(--spacing-1)',
266
+ fontStyle: 'italic',
267
+ }}
268
+ >
269
+ {run.errorMessage}
270
+ </div>
271
+ )}
272
+ </td>
273
+ <td
274
+ data-testid={`run-metrics-${run.runId}`}
275
+ style={{
276
+ padding: 'var(--space-component-sm) var(--space-component-md)',
277
+ }}
278
+ >
279
+ <div
280
+ style={{
281
+ fontSize: 'var(--font-size-overline)',
282
+ fontFamily: 'var(--font-family-mono)',
283
+ color: 'var(--color-text-muted)',
284
+ }}
285
+ >
286
+ {run.metrics.tokensUsed !== undefined && (
287
+ <div>Tokens: {run.metrics.tokensUsed}</div>
288
+ )}
289
+ {run.metrics.costUsd !== undefined && (
290
+ <div>Cost: ${run.metrics.costUsd.toFixed(4)}</div>
291
+ )}
292
+ {run.metrics.artifactsProduced !== undefined && (
293
+ <div>Artifacts: {run.metrics.artifactsProduced}</div>
294
+ )}
295
+ {Object.keys(run.metrics).length === 0 && <div>-</div>}
296
+ </div>
297
+ </td>
298
+ </tr>
299
+ ))}
300
+ </tbody>
301
+ </table>
302
+ )}
303
+ </div>
304
+ );
305
+ };
306
+
307
+ export default RunStatusTab;