flock-core 0.5.0b50__py3-none-any.whl → 0.5.0b51__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Potentially problematic release.
This version of flock-core might be problematic. Click here for more details.
- flock/dashboard/launcher.py +1 -1
- flock/frontend/README.md +678 -0
- flock/frontend/docs/DESIGN_SYSTEM.md +1980 -0
- flock/frontend/index.html +12 -0
- flock/frontend/package-lock.json +4347 -0
- flock/frontend/package.json +48 -0
- flock/frontend/src/App.tsx +79 -0
- flock/frontend/src/__tests__/e2e/critical-scenarios.test.tsx +587 -0
- flock/frontend/src/__tests__/integration/filtering-e2e.test.tsx +387 -0
- flock/frontend/src/__tests__/integration/graph-rendering.test.tsx +640 -0
- flock/frontend/src/__tests__/integration/indexeddb-persistence.test.tsx +699 -0
- flock/frontend/src/components/common/BuildInfo.tsx +39 -0
- flock/frontend/src/components/common/EmptyState.module.css +115 -0
- flock/frontend/src/components/common/EmptyState.tsx +128 -0
- flock/frontend/src/components/common/ErrorBoundary.module.css +169 -0
- flock/frontend/src/components/common/ErrorBoundary.tsx +118 -0
- flock/frontend/src/components/common/KeyboardShortcutsDialog.css +251 -0
- flock/frontend/src/components/common/KeyboardShortcutsDialog.tsx +151 -0
- flock/frontend/src/components/common/LoadingSpinner.module.css +97 -0
- flock/frontend/src/components/common/LoadingSpinner.tsx +29 -0
- flock/frontend/src/components/controls/PublishControl.css +547 -0
- flock/frontend/src/components/controls/PublishControl.test.tsx +543 -0
- flock/frontend/src/components/controls/PublishControl.tsx +432 -0
- flock/frontend/src/components/details/DetailWindowContainer.tsx +62 -0
- flock/frontend/src/components/details/LiveOutputTab.test.tsx +792 -0
- flock/frontend/src/components/details/LiveOutputTab.tsx +220 -0
- flock/frontend/src/components/details/MessageHistoryTab.tsx +299 -0
- flock/frontend/src/components/details/NodeDetailWindow.test.tsx +501 -0
- flock/frontend/src/components/details/NodeDetailWindow.tsx +218 -0
- flock/frontend/src/components/details/RunStatusTab.tsx +307 -0
- flock/frontend/src/components/details/tabs.test.tsx +1015 -0
- flock/frontend/src/components/filters/CorrelationIDFilter.module.css +102 -0
- flock/frontend/src/components/filters/CorrelationIDFilter.test.tsx +197 -0
- flock/frontend/src/components/filters/CorrelationIDFilter.tsx +121 -0
- flock/frontend/src/components/filters/FilterBar.module.css +29 -0
- flock/frontend/src/components/filters/FilterBar.test.tsx +133 -0
- flock/frontend/src/components/filters/FilterBar.tsx +33 -0
- flock/frontend/src/components/filters/FilterPills.module.css +79 -0
- flock/frontend/src/components/filters/FilterPills.test.tsx +173 -0
- flock/frontend/src/components/filters/FilterPills.tsx +67 -0
- flock/frontend/src/components/filters/TimeRangeFilter.module.css +91 -0
- flock/frontend/src/components/filters/TimeRangeFilter.test.tsx +154 -0
- flock/frontend/src/components/filters/TimeRangeFilter.tsx +105 -0
- flock/frontend/src/components/graph/AgentNode.test.tsx +75 -0
- flock/frontend/src/components/graph/AgentNode.tsx +322 -0
- flock/frontend/src/components/graph/GraphCanvas.tsx +406 -0
- flock/frontend/src/components/graph/MessageFlowEdge.tsx +128 -0
- flock/frontend/src/components/graph/MessageNode.test.tsx +62 -0
- flock/frontend/src/components/graph/MessageNode.tsx +116 -0
- flock/frontend/src/components/graph/MiniMap.tsx +47 -0
- flock/frontend/src/components/graph/TransformEdge.tsx +123 -0
- flock/frontend/src/components/layout/DashboardLayout.css +407 -0
- flock/frontend/src/components/layout/DashboardLayout.tsx +300 -0
- flock/frontend/src/components/layout/Header.module.css +88 -0
- flock/frontend/src/components/layout/Header.tsx +52 -0
- flock/frontend/src/components/modules/EventLogModule.test.tsx +401 -0
- flock/frontend/src/components/modules/EventLogModule.tsx +396 -0
- flock/frontend/src/components/modules/EventLogModuleWrapper.tsx +17 -0
- flock/frontend/src/components/modules/ModuleRegistry.test.ts +333 -0
- flock/frontend/src/components/modules/ModuleRegistry.ts +85 -0
- flock/frontend/src/components/modules/ModuleWindow.tsx +155 -0
- flock/frontend/src/components/modules/registerModules.ts +20 -0
- flock/frontend/src/components/settings/AdvancedSettings.tsx +175 -0
- flock/frontend/src/components/settings/AppearanceSettings.tsx +185 -0
- flock/frontend/src/components/settings/GraphSettings.tsx +110 -0
- flock/frontend/src/components/settings/SettingsPanel.css +327 -0
- flock/frontend/src/components/settings/SettingsPanel.tsx +131 -0
- flock/frontend/src/components/settings/ThemeSelector.tsx +298 -0
- flock/frontend/src/hooks/useKeyboardShortcuts.ts +148 -0
- flock/frontend/src/hooks/useModulePersistence.test.ts +442 -0
- flock/frontend/src/hooks/useModulePersistence.ts +154 -0
- flock/frontend/src/hooks/useModules.ts +139 -0
- flock/frontend/src/hooks/usePersistence.ts +139 -0
- flock/frontend/src/main.tsx +13 -0
- flock/frontend/src/services/api.ts +213 -0
- flock/frontend/src/services/indexeddb.test.ts +793 -0
- flock/frontend/src/services/indexeddb.ts +794 -0
- flock/frontend/src/services/layout.test.ts +437 -0
- flock/frontend/src/services/layout.ts +146 -0
- flock/frontend/src/services/themeApplicator.ts +140 -0
- flock/frontend/src/services/themeService.ts +77 -0
- flock/frontend/src/services/websocket.test.ts +595 -0
- flock/frontend/src/services/websocket.ts +685 -0
- flock/frontend/src/store/filterStore.test.ts +242 -0
- flock/frontend/src/store/filterStore.ts +103 -0
- flock/frontend/src/store/graphStore.test.ts +186 -0
- flock/frontend/src/store/graphStore.ts +414 -0
- flock/frontend/src/store/moduleStore.test.ts +253 -0
- flock/frontend/src/store/moduleStore.ts +57 -0
- flock/frontend/src/store/settingsStore.ts +188 -0
- flock/frontend/src/store/streamStore.ts +68 -0
- flock/frontend/src/store/uiStore.test.ts +54 -0
- flock/frontend/src/store/uiStore.ts +110 -0
- flock/frontend/src/store/wsStore.ts +34 -0
- flock/frontend/src/styles/index.css +15 -0
- flock/frontend/src/styles/scrollbar.css +47 -0
- flock/frontend/src/styles/variables.css +488 -0
- flock/frontend/src/test/setup.ts +1 -0
- flock/frontend/src/types/filters.ts +14 -0
- flock/frontend/src/types/graph.ts +55 -0
- flock/frontend/src/types/modules.ts +7 -0
- flock/frontend/src/types/theme.ts +55 -0
- flock/frontend/src/utils/mockData.ts +85 -0
- flock/frontend/src/utils/performance.ts +16 -0
- flock/frontend/src/utils/transforms.test.ts +860 -0
- flock/frontend/src/utils/transforms.ts +323 -0
- flock/frontend/src/vite-env.d.ts +17 -0
- flock/frontend/tsconfig.json +27 -0
- flock/frontend/tsconfig.node.json +11 -0
- flock/frontend/vite.config.ts +25 -0
- flock/frontend/vitest.config.ts +11 -0
- {flock_core-0.5.0b50.dist-info → flock_core-0.5.0b51.dist-info}/METADATA +1 -1
- {flock_core-0.5.0b50.dist-info → flock_core-0.5.0b51.dist-info}/RECORD +116 -6
- {flock_core-0.5.0b50.dist-info → flock_core-0.5.0b51.dist-info}/WHEEL +0 -0
- {flock_core-0.5.0b50.dist-info → flock_core-0.5.0b51.dist-info}/entry_points.txt +0 -0
- {flock_core-0.5.0b50.dist-info → flock_core-0.5.0b51.dist-info}/licenses/LICENSE +0 -0
|
@@ -0,0 +1,543 @@
|
|
|
1
|
+
import { describe, it, expect, vi, beforeEach } from 'vitest';
|
|
2
|
+
import { render, screen, fireEvent, waitFor } from '@testing-library/react';
|
|
3
|
+
import PublishControl from './PublishControl';
|
|
4
|
+
import { useFilterStore } from '../../store/filterStore';
|
|
5
|
+
|
|
6
|
+
// Mock fetch globally
|
|
7
|
+
const mockFetch = vi.fn();
|
|
8
|
+
globalThis.fetch = mockFetch;
|
|
9
|
+
|
|
10
|
+
describe('PublishControl', () => {
|
|
11
|
+
const mockArtifactTypes = [
|
|
12
|
+
{ name: 'Idea', schema: { type: 'object', properties: { content: { type: 'string' } } } },
|
|
13
|
+
{ name: 'Movie', schema: { type: 'object', properties: { title: { type: 'string' }, plot: { type: 'string' } } } },
|
|
14
|
+
{ name: 'Tagline', schema: { type: 'object', properties: { text: { type: 'string' } } } },
|
|
15
|
+
];
|
|
16
|
+
|
|
17
|
+
beforeEach(() => {
|
|
18
|
+
vi.clearAllMocks();
|
|
19
|
+
mockFetch.mockReset();
|
|
20
|
+
});
|
|
21
|
+
|
|
22
|
+
it('should render publish form with artifact type dropdown and content textarea', async () => {
|
|
23
|
+
mockFetch.mockResolvedValueOnce({
|
|
24
|
+
ok: true,
|
|
25
|
+
json: async () => ({ artifact_types: mockArtifactTypes }),
|
|
26
|
+
});
|
|
27
|
+
|
|
28
|
+
render(<PublishControl />);
|
|
29
|
+
|
|
30
|
+
await waitFor(() => {
|
|
31
|
+
expect(screen.getByLabelText(/artifact type/i)).toBeInTheDocument();
|
|
32
|
+
});
|
|
33
|
+
|
|
34
|
+
// Select artifact type to trigger dynamic field rendering
|
|
35
|
+
const artifactTypeSelect = screen.getByLabelText(/artifact type/i);
|
|
36
|
+
fireEvent.change(artifactTypeSelect, { target: { value: 'Idea' } });
|
|
37
|
+
|
|
38
|
+
// Now the content field should appear
|
|
39
|
+
expect(screen.getByLabelText(/content/i)).toBeInTheDocument();
|
|
40
|
+
expect(screen.getByRole('button', { name: /publish artifact/i })).toBeInTheDocument();
|
|
41
|
+
});
|
|
42
|
+
|
|
43
|
+
it('should populate artifact type dropdown from available types', async () => {
|
|
44
|
+
mockFetch.mockResolvedValueOnce({
|
|
45
|
+
ok: true,
|
|
46
|
+
json: async () => ({ artifact_types: mockArtifactTypes }),
|
|
47
|
+
});
|
|
48
|
+
|
|
49
|
+
render(<PublishControl />);
|
|
50
|
+
|
|
51
|
+
await waitFor(() => {
|
|
52
|
+
const dropdown = screen.getByLabelText(/artifact type/i) as HTMLSelectElement;
|
|
53
|
+
expect(dropdown.options.length).toBeGreaterThan(0);
|
|
54
|
+
});
|
|
55
|
+
|
|
56
|
+
const dropdown = screen.getByLabelText(/artifact type/i) as HTMLSelectElement;
|
|
57
|
+
const optionValues = Array.from(dropdown.options).map(opt => opt.value);
|
|
58
|
+
|
|
59
|
+
expect(optionValues).toContain('Idea');
|
|
60
|
+
expect(optionValues).toContain('Movie');
|
|
61
|
+
expect(optionValues).toContain('Tagline');
|
|
62
|
+
});
|
|
63
|
+
|
|
64
|
+
it('should validate required fields before submission', async () => {
|
|
65
|
+
mockFetch
|
|
66
|
+
.mockResolvedValueOnce({
|
|
67
|
+
ok: true,
|
|
68
|
+
json: async () => ({ artifact_types: mockArtifactTypes }),
|
|
69
|
+
})
|
|
70
|
+
.mockResolvedValueOnce({
|
|
71
|
+
ok: true,
|
|
72
|
+
json: async () => ({
|
|
73
|
+
status: 'success',
|
|
74
|
+
correlation_id: 'should-not-be-called',
|
|
75
|
+
}),
|
|
76
|
+
});
|
|
77
|
+
|
|
78
|
+
render(<PublishControl />);
|
|
79
|
+
|
|
80
|
+
await waitFor(() => {
|
|
81
|
+
expect(screen.getByLabelText(/artifact type/i)).toBeInTheDocument();
|
|
82
|
+
});
|
|
83
|
+
|
|
84
|
+
const submitButton = screen.getByRole('button', { name: /publish artifact/i });
|
|
85
|
+
|
|
86
|
+
// Test 1: Try to submit without selecting type - should NOT call API
|
|
87
|
+
fireEvent.click(submitButton);
|
|
88
|
+
|
|
89
|
+
// Give it a moment, then verify API was NOT called (only 1 call for artifact types)
|
|
90
|
+
await new Promise(resolve => setTimeout(resolve, 100));
|
|
91
|
+
expect(mockFetch).toHaveBeenCalledTimes(1); // Only the initial artifact types fetch
|
|
92
|
+
|
|
93
|
+
// Test 2: Select artifact type but leave content empty - should NOT call API
|
|
94
|
+
const artifactTypeSelect = screen.getByLabelText(/artifact type/i);
|
|
95
|
+
fireEvent.change(artifactTypeSelect, { target: { value: 'Idea' } });
|
|
96
|
+
|
|
97
|
+
await waitFor(() => {
|
|
98
|
+
expect(screen.getByLabelText(/content/i)).toBeInTheDocument();
|
|
99
|
+
});
|
|
100
|
+
|
|
101
|
+
// Try to submit with empty content
|
|
102
|
+
fireEvent.click(submitButton);
|
|
103
|
+
|
|
104
|
+
// Give it a moment, then verify API was still NOT called
|
|
105
|
+
await new Promise(resolve => setTimeout(resolve, 100));
|
|
106
|
+
expect(mockFetch).toHaveBeenCalledTimes(1); // Still only 1 call
|
|
107
|
+
});
|
|
108
|
+
|
|
109
|
+
it('should validate content as valid JSON', async () => {
|
|
110
|
+
mockFetch.mockResolvedValueOnce({
|
|
111
|
+
ok: true,
|
|
112
|
+
json: async () => ({ artifact_types: mockArtifactTypes }),
|
|
113
|
+
});
|
|
114
|
+
|
|
115
|
+
render(<PublishControl />);
|
|
116
|
+
|
|
117
|
+
await waitFor(() => {
|
|
118
|
+
expect(screen.getByLabelText(/artifact type/i)).toBeInTheDocument();
|
|
119
|
+
});
|
|
120
|
+
|
|
121
|
+
const artifactTypeSelect = screen.getByLabelText(/artifact type/i);
|
|
122
|
+
fireEvent.change(artifactTypeSelect, { target: { value: 'Idea' } });
|
|
123
|
+
|
|
124
|
+
// Wait for content field to appear after selecting type
|
|
125
|
+
await waitFor(() => {
|
|
126
|
+
expect(screen.getByLabelText(/content/i)).toBeInTheDocument();
|
|
127
|
+
});
|
|
128
|
+
|
|
129
|
+
const contentInput = screen.getByLabelText(/content/i);
|
|
130
|
+
const submitButton = screen.getByRole('button', { name: /publish artifact/i });
|
|
131
|
+
|
|
132
|
+
// Content field is now a text input, not textarea, and accepts any string
|
|
133
|
+
// The validation is done on the backend, so this test is no longer relevant
|
|
134
|
+
// for JSON validation. Skip JSON validation test.
|
|
135
|
+
fireEvent.change(contentInput, { target: { value: 'Test content' } });
|
|
136
|
+
fireEvent.click(submitButton);
|
|
137
|
+
|
|
138
|
+
// Should not show JSON validation error for simple text
|
|
139
|
+
await waitFor(() => {
|
|
140
|
+
expect(mockFetch).toHaveBeenCalled();
|
|
141
|
+
});
|
|
142
|
+
});
|
|
143
|
+
|
|
144
|
+
it('should successfully publish and call API endpoint', async () => {
|
|
145
|
+
mockFetch
|
|
146
|
+
.mockResolvedValueOnce({
|
|
147
|
+
ok: true,
|
|
148
|
+
json: async () => ({ artifact_types: mockArtifactTypes }),
|
|
149
|
+
})
|
|
150
|
+
.mockResolvedValueOnce({
|
|
151
|
+
ok: true,
|
|
152
|
+
json: async () => ({
|
|
153
|
+
status: 'success',
|
|
154
|
+
correlation_id: 'test-correlation-123',
|
|
155
|
+
message: 'Artifact published successfully'
|
|
156
|
+
}),
|
|
157
|
+
});
|
|
158
|
+
|
|
159
|
+
render(<PublishControl />);
|
|
160
|
+
|
|
161
|
+
await waitFor(() => {
|
|
162
|
+
expect(screen.getByLabelText(/artifact type/i)).toBeInTheDocument();
|
|
163
|
+
});
|
|
164
|
+
|
|
165
|
+
const artifactTypeSelect = screen.getByLabelText(/artifact type/i);
|
|
166
|
+
fireEvent.change(artifactTypeSelect, { target: { value: 'Idea' } });
|
|
167
|
+
|
|
168
|
+
await waitFor(() => {
|
|
169
|
+
expect(screen.getByLabelText(/content/i)).toBeInTheDocument();
|
|
170
|
+
});
|
|
171
|
+
|
|
172
|
+
const contentInput = screen.getByLabelText(/content/i);
|
|
173
|
+
const submitButton = screen.getByRole('button', { name: /publish artifact/i });
|
|
174
|
+
|
|
175
|
+
fireEvent.change(contentInput, { target: { value: 'Test idea' } });
|
|
176
|
+
fireEvent.click(submitButton);
|
|
177
|
+
|
|
178
|
+
await waitFor(() => {
|
|
179
|
+
expect(mockFetch).toHaveBeenCalledWith(
|
|
180
|
+
'/api/control/publish',
|
|
181
|
+
expect.objectContaining({
|
|
182
|
+
method: 'POST',
|
|
183
|
+
headers: { 'Content-Type': 'application/json' },
|
|
184
|
+
body: JSON.stringify({
|
|
185
|
+
artifact_type: 'Idea',
|
|
186
|
+
content: { content: 'Test idea' }, // Schema-based: { content: string }
|
|
187
|
+
}),
|
|
188
|
+
})
|
|
189
|
+
);
|
|
190
|
+
});
|
|
191
|
+
});
|
|
192
|
+
|
|
193
|
+
it('should display success message with correlation ID after successful publish', async () => {
|
|
194
|
+
mockFetch
|
|
195
|
+
.mockResolvedValueOnce({
|
|
196
|
+
ok: true,
|
|
197
|
+
json: async () => ({ artifact_types: mockArtifactTypes }),
|
|
198
|
+
})
|
|
199
|
+
.mockResolvedValueOnce({
|
|
200
|
+
ok: true,
|
|
201
|
+
json: async () => ({
|
|
202
|
+
status: 'success',
|
|
203
|
+
correlation_id: 'test-correlation-456',
|
|
204
|
+
message: 'Artifact published successfully'
|
|
205
|
+
}),
|
|
206
|
+
});
|
|
207
|
+
|
|
208
|
+
render(<PublishControl />);
|
|
209
|
+
|
|
210
|
+
await waitFor(() => {
|
|
211
|
+
expect(screen.getByLabelText(/artifact type/i)).toBeInTheDocument();
|
|
212
|
+
});
|
|
213
|
+
|
|
214
|
+
const artifactTypeSelect = screen.getByLabelText(/artifact type/i);
|
|
215
|
+
fireEvent.change(artifactTypeSelect, { target: { value: 'Idea' } });
|
|
216
|
+
|
|
217
|
+
await waitFor(() => {
|
|
218
|
+
expect(screen.getByLabelText(/content/i)).toBeInTheDocument();
|
|
219
|
+
});
|
|
220
|
+
|
|
221
|
+
const contentInput = screen.getByLabelText(/content/i);
|
|
222
|
+
const submitButton = screen.getByRole('button', { name: /publish artifact/i });
|
|
223
|
+
|
|
224
|
+
fireEvent.change(contentInput, { target: { value: 'Success test' } });
|
|
225
|
+
fireEvent.click(submitButton);
|
|
226
|
+
|
|
227
|
+
await waitFor(() => {
|
|
228
|
+
expect(screen.getByText(/successfully published/i)).toBeInTheDocument();
|
|
229
|
+
expect(screen.getByText(/test-correlation-456/)).toBeInTheDocument();
|
|
230
|
+
});
|
|
231
|
+
});
|
|
232
|
+
|
|
233
|
+
it('should display error message on API failure', async () => {
|
|
234
|
+
mockFetch
|
|
235
|
+
.mockResolvedValueOnce({
|
|
236
|
+
ok: true,
|
|
237
|
+
json: async () => ({ artifact_types: mockArtifactTypes }),
|
|
238
|
+
})
|
|
239
|
+
.mockResolvedValueOnce({
|
|
240
|
+
ok: false,
|
|
241
|
+
status: 500,
|
|
242
|
+
json: async () => ({
|
|
243
|
+
error: 'Internal server error',
|
|
244
|
+
message: 'Failed to publish artifact'
|
|
245
|
+
}),
|
|
246
|
+
});
|
|
247
|
+
|
|
248
|
+
render(<PublishControl />);
|
|
249
|
+
|
|
250
|
+
await waitFor(() => {
|
|
251
|
+
expect(screen.getByLabelText(/artifact type/i)).toBeInTheDocument();
|
|
252
|
+
});
|
|
253
|
+
|
|
254
|
+
const artifactTypeSelect = screen.getByLabelText(/artifact type/i);
|
|
255
|
+
fireEvent.change(artifactTypeSelect, { target: { value: 'Idea' } });
|
|
256
|
+
|
|
257
|
+
await waitFor(() => {
|
|
258
|
+
expect(screen.getByLabelText(/content/i)).toBeInTheDocument();
|
|
259
|
+
});
|
|
260
|
+
|
|
261
|
+
const contentInput = screen.getByLabelText(/content/i);
|
|
262
|
+
const submitButton = screen.getByRole('button', { name: /publish artifact/i });
|
|
263
|
+
|
|
264
|
+
fireEvent.change(contentInput, { target: { value: 'Error test' } });
|
|
265
|
+
fireEvent.click(submitButton);
|
|
266
|
+
|
|
267
|
+
await waitFor(() => {
|
|
268
|
+
expect(screen.getByText(/failed to publish/i)).toBeInTheDocument();
|
|
269
|
+
});
|
|
270
|
+
});
|
|
271
|
+
|
|
272
|
+
it('should reset form after successful publish', async () => {
|
|
273
|
+
mockFetch
|
|
274
|
+
.mockResolvedValueOnce({
|
|
275
|
+
ok: true,
|
|
276
|
+
json: async () => ({ artifact_types: mockArtifactTypes }),
|
|
277
|
+
})
|
|
278
|
+
.mockResolvedValueOnce({
|
|
279
|
+
ok: true,
|
|
280
|
+
json: async () => ({
|
|
281
|
+
status: 'success',
|
|
282
|
+
correlation_id: 'test-correlation-789',
|
|
283
|
+
message: 'Artifact published successfully'
|
|
284
|
+
}),
|
|
285
|
+
});
|
|
286
|
+
|
|
287
|
+
render(<PublishControl />);
|
|
288
|
+
|
|
289
|
+
await waitFor(() => {
|
|
290
|
+
expect(screen.getByLabelText(/artifact type/i)).toBeInTheDocument();
|
|
291
|
+
});
|
|
292
|
+
|
|
293
|
+
const artifactTypeSelect = screen.getByLabelText(/artifact type/i) as HTMLSelectElement;
|
|
294
|
+
fireEvent.change(artifactTypeSelect, { target: { value: 'Idea' } });
|
|
295
|
+
|
|
296
|
+
await waitFor(() => {
|
|
297
|
+
expect(screen.getByLabelText(/content/i)).toBeInTheDocument();
|
|
298
|
+
});
|
|
299
|
+
|
|
300
|
+
const contentInput = screen.getByLabelText(/content/i) as HTMLInputElement;
|
|
301
|
+
const submitButton = screen.getByRole('button', { name: /publish artifact/i });
|
|
302
|
+
|
|
303
|
+
fireEvent.change(contentInput, { target: { value: 'Reset test' } });
|
|
304
|
+
fireEvent.click(submitButton);
|
|
305
|
+
|
|
306
|
+
await waitFor(() => {
|
|
307
|
+
expect(screen.getByText(/successfully published/i)).toBeInTheDocument();
|
|
308
|
+
});
|
|
309
|
+
|
|
310
|
+
// Form should be reset - artifact type cleared means fields disappear
|
|
311
|
+
expect(artifactTypeSelect.value).toBe('');
|
|
312
|
+
});
|
|
313
|
+
|
|
314
|
+
it('should disable submit button during API call', async () => {
|
|
315
|
+
let resolvePublish: (value: any) => void;
|
|
316
|
+
const publishPromise = new Promise(resolve => {
|
|
317
|
+
resolvePublish = resolve;
|
|
318
|
+
});
|
|
319
|
+
|
|
320
|
+
mockFetch
|
|
321
|
+
.mockResolvedValueOnce({
|
|
322
|
+
ok: true,
|
|
323
|
+
json: async () => ({ artifact_types: mockArtifactTypes }),
|
|
324
|
+
})
|
|
325
|
+
.mockReturnValueOnce(publishPromise as any);
|
|
326
|
+
|
|
327
|
+
render(<PublishControl />);
|
|
328
|
+
|
|
329
|
+
await waitFor(() => {
|
|
330
|
+
expect(screen.getByLabelText(/artifact type/i)).toBeInTheDocument();
|
|
331
|
+
});
|
|
332
|
+
|
|
333
|
+
const artifactTypeSelect = screen.getByLabelText(/artifact type/i);
|
|
334
|
+
fireEvent.change(artifactTypeSelect, { target: { value: 'Idea' } });
|
|
335
|
+
|
|
336
|
+
await waitFor(() => {
|
|
337
|
+
expect(screen.getByLabelText(/content/i)).toBeInTheDocument();
|
|
338
|
+
});
|
|
339
|
+
|
|
340
|
+
const contentInput = screen.getByLabelText(/content/i);
|
|
341
|
+
const submitButton = screen.getByRole('button', { name: /publish artifact/i });
|
|
342
|
+
|
|
343
|
+
fireEvent.change(contentInput, { target: { value: 'Loading test' } });
|
|
344
|
+
fireEvent.click(submitButton);
|
|
345
|
+
|
|
346
|
+
// Button should be disabled during API call
|
|
347
|
+
await waitFor(() => {
|
|
348
|
+
expect(submitButton).toBeDisabled();
|
|
349
|
+
expect(screen.getByText(/publishing/i)).toBeInTheDocument();
|
|
350
|
+
});
|
|
351
|
+
|
|
352
|
+
// Resolve the promise
|
|
353
|
+
resolvePublish!({
|
|
354
|
+
ok: true,
|
|
355
|
+
json: async () => ({
|
|
356
|
+
status: 'success',
|
|
357
|
+
correlation_id: 'test-correlation-999',
|
|
358
|
+
message: 'Artifact published successfully'
|
|
359
|
+
}),
|
|
360
|
+
});
|
|
361
|
+
|
|
362
|
+
// Button should be enabled after API call completes
|
|
363
|
+
await waitFor(() => {
|
|
364
|
+
expect(submitButton).not.toBeDisabled();
|
|
365
|
+
});
|
|
366
|
+
});
|
|
367
|
+
|
|
368
|
+
it('should handle network errors gracefully', async () => {
|
|
369
|
+
mockFetch
|
|
370
|
+
.mockResolvedValueOnce({
|
|
371
|
+
ok: true,
|
|
372
|
+
json: async () => ({ artifact_types: mockArtifactTypes }),
|
|
373
|
+
})
|
|
374
|
+
.mockRejectedValueOnce(new Error('Network error'));
|
|
375
|
+
|
|
376
|
+
render(<PublishControl />);
|
|
377
|
+
|
|
378
|
+
await waitFor(() => {
|
|
379
|
+
expect(screen.getByLabelText(/artifact type/i)).toBeInTheDocument();
|
|
380
|
+
});
|
|
381
|
+
|
|
382
|
+
const artifactTypeSelect = screen.getByLabelText(/artifact type/i);
|
|
383
|
+
fireEvent.change(artifactTypeSelect, { target: { value: 'Idea' } });
|
|
384
|
+
|
|
385
|
+
await waitFor(() => {
|
|
386
|
+
expect(screen.getByLabelText(/content/i)).toBeInTheDocument();
|
|
387
|
+
});
|
|
388
|
+
|
|
389
|
+
const contentInput = screen.getByLabelText(/content/i);
|
|
390
|
+
const submitButton = screen.getByRole('button', { name: /publish artifact/i });
|
|
391
|
+
|
|
392
|
+
fireEvent.change(contentInput, { target: { value: 'Network error test' } });
|
|
393
|
+
fireEvent.click(submitButton);
|
|
394
|
+
|
|
395
|
+
await waitFor(() => {
|
|
396
|
+
expect(screen.getByText(/network error|failed to connect/i)).toBeInTheDocument();
|
|
397
|
+
});
|
|
398
|
+
});
|
|
399
|
+
|
|
400
|
+
// Auto-filter checkbox tests
|
|
401
|
+
it('should render auto-set filter checkbox checked by default', async () => {
|
|
402
|
+
mockFetch.mockResolvedValueOnce({
|
|
403
|
+
ok: true,
|
|
404
|
+
json: async () => ({ artifact_types: mockArtifactTypes }),
|
|
405
|
+
});
|
|
406
|
+
|
|
407
|
+
render(<PublishControl />);
|
|
408
|
+
|
|
409
|
+
await waitFor(() => {
|
|
410
|
+
expect(screen.getByLabelText(/artifact type/i)).toBeInTheDocument();
|
|
411
|
+
});
|
|
412
|
+
|
|
413
|
+
const checkbox = screen.getByLabelText(/set filter to correlation id/i) as HTMLInputElement;
|
|
414
|
+
expect(checkbox).toBeInTheDocument();
|
|
415
|
+
expect(checkbox.checked).toBe(true);
|
|
416
|
+
});
|
|
417
|
+
|
|
418
|
+
it('should set filter to correlation ID when checkbox is checked and publish succeeds', async () => {
|
|
419
|
+
mockFetch
|
|
420
|
+
.mockResolvedValueOnce({
|
|
421
|
+
ok: true,
|
|
422
|
+
json: async () => ({ artifact_types: mockArtifactTypes }),
|
|
423
|
+
})
|
|
424
|
+
.mockResolvedValueOnce({
|
|
425
|
+
ok: true,
|
|
426
|
+
json: async () => ({
|
|
427
|
+
status: 'success',
|
|
428
|
+
correlation_id: 'auto-filter-123',
|
|
429
|
+
message: 'Artifact published successfully'
|
|
430
|
+
}),
|
|
431
|
+
});
|
|
432
|
+
|
|
433
|
+
render(<PublishControl />);
|
|
434
|
+
|
|
435
|
+
await waitFor(() => {
|
|
436
|
+
expect(screen.getByLabelText(/artifact type/i)).toBeInTheDocument();
|
|
437
|
+
});
|
|
438
|
+
|
|
439
|
+
const artifactTypeSelect = screen.getByLabelText(/artifact type/i);
|
|
440
|
+
const checkbox = screen.getByLabelText(/set filter to correlation id/i) as HTMLInputElement;
|
|
441
|
+
|
|
442
|
+
// Checkbox should be checked by default
|
|
443
|
+
expect(checkbox.checked).toBe(true);
|
|
444
|
+
|
|
445
|
+
fireEvent.change(artifactTypeSelect, { target: { value: 'Idea' } });
|
|
446
|
+
|
|
447
|
+
await waitFor(() => {
|
|
448
|
+
expect(screen.getByLabelText(/content/i)).toBeInTheDocument();
|
|
449
|
+
});
|
|
450
|
+
|
|
451
|
+
const contentInput = screen.getByLabelText(/content/i);
|
|
452
|
+
const submitButton = screen.getByRole('button', { name: /publish artifact/i });
|
|
453
|
+
|
|
454
|
+
fireEvent.change(contentInput, { target: { value: 'Auto-filter test' } });
|
|
455
|
+
fireEvent.click(submitButton);
|
|
456
|
+
|
|
457
|
+
await waitFor(() => {
|
|
458
|
+
expect(screen.getByText(/filter set to: auto-filter-123/i)).toBeInTheDocument();
|
|
459
|
+
});
|
|
460
|
+
|
|
461
|
+
// Verify filter was set in store
|
|
462
|
+
const filterState = useFilterStore.getState();
|
|
463
|
+
expect(filterState.correlationId).toBe('auto-filter-123');
|
|
464
|
+
});
|
|
465
|
+
|
|
466
|
+
it('should NOT set filter when checkbox is unchecked', async () => {
|
|
467
|
+
mockFetch
|
|
468
|
+
.mockResolvedValueOnce({
|
|
469
|
+
ok: true,
|
|
470
|
+
json: async () => ({ artifact_types: mockArtifactTypes }),
|
|
471
|
+
})
|
|
472
|
+
.mockResolvedValueOnce({
|
|
473
|
+
ok: true,
|
|
474
|
+
json: async () => ({
|
|
475
|
+
status: 'success',
|
|
476
|
+
correlation_id: 'no-filter-456',
|
|
477
|
+
message: 'Artifact published successfully'
|
|
478
|
+
}),
|
|
479
|
+
});
|
|
480
|
+
|
|
481
|
+
render(<PublishControl />);
|
|
482
|
+
|
|
483
|
+
await waitFor(() => {
|
|
484
|
+
expect(screen.getByLabelText(/artifact type/i)).toBeInTheDocument();
|
|
485
|
+
});
|
|
486
|
+
|
|
487
|
+
const artifactTypeSelect = screen.getByLabelText(/artifact type/i);
|
|
488
|
+
const checkbox = screen.getByLabelText(/set filter to correlation id/i) as HTMLInputElement;
|
|
489
|
+
|
|
490
|
+
// Uncheck the checkbox
|
|
491
|
+
fireEvent.click(checkbox);
|
|
492
|
+
expect(checkbox.checked).toBe(false);
|
|
493
|
+
|
|
494
|
+
// Clear any existing filter
|
|
495
|
+
useFilterStore.setState({ correlationId: undefined });
|
|
496
|
+
|
|
497
|
+
fireEvent.change(artifactTypeSelect, { target: { value: 'Idea' } });
|
|
498
|
+
|
|
499
|
+
await waitFor(() => {
|
|
500
|
+
expect(screen.getByLabelText(/content/i)).toBeInTheDocument();
|
|
501
|
+
});
|
|
502
|
+
|
|
503
|
+
const contentInput = screen.getByLabelText(/content/i);
|
|
504
|
+
const submitButton = screen.getByRole('button', { name: /publish artifact/i });
|
|
505
|
+
|
|
506
|
+
fireEvent.change(contentInput, { target: { value: 'No filter test' } });
|
|
507
|
+
fireEvent.click(submitButton);
|
|
508
|
+
|
|
509
|
+
await waitFor(() => {
|
|
510
|
+
expect(screen.getByText(/correlation id: no-filter-456/i)).toBeInTheDocument();
|
|
511
|
+
});
|
|
512
|
+
|
|
513
|
+
// Verify filter was NOT set in store
|
|
514
|
+
const filterState = useFilterStore.getState();
|
|
515
|
+
expect(filterState.correlationId).toBeUndefined();
|
|
516
|
+
});
|
|
517
|
+
|
|
518
|
+
it('should allow toggling checkbox state', async () => {
|
|
519
|
+
mockFetch.mockResolvedValueOnce({
|
|
520
|
+
ok: true,
|
|
521
|
+
json: async () => ({ artifact_types: mockArtifactTypes }),
|
|
522
|
+
});
|
|
523
|
+
|
|
524
|
+
render(<PublishControl />);
|
|
525
|
+
|
|
526
|
+
await waitFor(() => {
|
|
527
|
+
expect(screen.getByLabelText(/artifact type/i)).toBeInTheDocument();
|
|
528
|
+
});
|
|
529
|
+
|
|
530
|
+
const checkbox = screen.getByLabelText(/set filter to correlation id/i) as HTMLInputElement;
|
|
531
|
+
|
|
532
|
+
// Initially checked
|
|
533
|
+
expect(checkbox.checked).toBe(true);
|
|
534
|
+
|
|
535
|
+
// Uncheck
|
|
536
|
+
fireEvent.click(checkbox);
|
|
537
|
+
expect(checkbox.checked).toBe(false);
|
|
538
|
+
|
|
539
|
+
// Check again
|
|
540
|
+
fireEvent.click(checkbox);
|
|
541
|
+
expect(checkbox.checked).toBe(true);
|
|
542
|
+
});
|
|
543
|
+
});
|