test(admin-v2): Phase 2 frontend tests — 61 tests for 5 pages
Some checks failed
CI / Lint & TypeCheck (push) Has been cancelled
CI / Unit Tests (push) Has been cancelled
CI / Build Frontend (push) Has been cancelled
CI / Rust Check (push) Has been cancelled
CI / Security Scan (push) Has been cancelled
CI / E2E Tests (push) Has been cancelled
Some checks failed
CI / Lint & TypeCheck (push) Has been cancelled
CI / Unit Tests (push) Has been cancelled
CI / Build Frontend (push) Has been cancelled
CI / Rust Check (push) Has been cancelled
CI / Security Scan (push) Has been cancelled
CI / E2E Tests (push) Has been cancelled
- Billing (13 tests): plan cards, prices, limits, usage bars, payment flow - ScheduledTasks (16 tests): CRUD table, schedule/target types, color tags - Knowledge (12 tests): 4 tabs, items/categories/search/analytics panels - Roles (12 tests): roles + permission templates tabs - ConfigSync (8 tests): sync log viewer with action labels Fix: Knowledge.tsx missing </Select> and </Modal> closing tags (JSX parse error) Fix: tests/setup.ts added ResizeObserver mock for ProTable compatibility
This commit is contained in:
153
admin-v2/tests/pages/ConfigSync.test.tsx
Normal file
153
admin-v2/tests/pages/ConfigSync.test.tsx
Normal file
@@ -0,0 +1,153 @@
|
||||
import { describe, it, expect, beforeEach, afterEach } from 'vitest'
|
||||
import { render, screen, waitFor } from '@testing-library/react'
|
||||
import { http, HttpResponse } from 'msw'
|
||||
import { setupServer } from 'msw/node'
|
||||
import { QueryClient, QueryClientProvider } from '@tanstack/react-query'
|
||||
|
||||
import ConfigSync from '@/pages/ConfigSync'
|
||||
|
||||
const mockSyncLogs = {
|
||||
items: [
|
||||
{
|
||||
id: 1,
|
||||
account_id: 'acc-001',
|
||||
client_fingerprint: 'fp-abc123def456',
|
||||
action: 'push',
|
||||
config_keys: 'model_config,prompt_config',
|
||||
client_values: '{"model":"gpt-4"}',
|
||||
saas_values: '{"model":"gpt-3.5"}',
|
||||
resolution: 'client_wins',
|
||||
created_at: '2026-04-07T10:30:00Z',
|
||||
},
|
||||
{
|
||||
id: 2,
|
||||
account_id: 'acc-002',
|
||||
client_fingerprint: 'fp-xyz789',
|
||||
action: 'pull',
|
||||
config_keys: 'privacy_settings',
|
||||
client_values: null,
|
||||
saas_values: '{"analytics":true}',
|
||||
resolution: null,
|
||||
created_at: '2026-04-06T08:00:00Z',
|
||||
},
|
||||
],
|
||||
total: 2,
|
||||
page: 1,
|
||||
page_size: 20,
|
||||
}
|
||||
|
||||
const server = setupServer()
|
||||
|
||||
beforeEach(() => {
|
||||
server.listen({ onUnhandledRequest: 'bypass' })
|
||||
})
|
||||
|
||||
afterEach(() => {
|
||||
server.close()
|
||||
})
|
||||
|
||||
function renderWithProviders(ui: React.ReactElement) {
|
||||
const queryClient = new QueryClient({
|
||||
defaultOptions: { queries: { retry: false } },
|
||||
})
|
||||
return render(
|
||||
<QueryClientProvider client={queryClient}>
|
||||
{ui}
|
||||
</QueryClientProvider>,
|
||||
)
|
||||
}
|
||||
|
||||
describe('ConfigSync', () => {
|
||||
it('renders page title', () => {
|
||||
server.use(
|
||||
http.get('*/api/v1/config/sync-logs', () => {
|
||||
return HttpResponse.json({ items: [], total: 0, page: 1, page_size: 20 })
|
||||
}),
|
||||
)
|
||||
renderWithProviders(<ConfigSync />)
|
||||
expect(screen.getByText('配置同步日志')).toBeInTheDocument()
|
||||
})
|
||||
|
||||
it('shows loading state', async () => {
|
||||
server.use(
|
||||
http.get('*/api/v1/config/sync-logs', async () => {
|
||||
await new Promise(resolve => setTimeout(resolve, 500))
|
||||
return HttpResponse.json(mockSyncLogs)
|
||||
}),
|
||||
)
|
||||
renderWithProviders(<ConfigSync />)
|
||||
expect(document.querySelector('.ant-spin')).toBeTruthy()
|
||||
})
|
||||
|
||||
it('displays sync logs with Chinese action labels', async () => {
|
||||
server.use(
|
||||
http.get('*/api/v1/config/sync-logs', () => {
|
||||
return HttpResponse.json(mockSyncLogs)
|
||||
}),
|
||||
)
|
||||
renderWithProviders(<ConfigSync />)
|
||||
|
||||
// Action labels are mapped to Chinese: push → 推送, pull → 拉取
|
||||
await waitFor(() => {
|
||||
expect(screen.getByText('推送')).toBeInTheDocument()
|
||||
})
|
||||
expect(screen.getByText('拉取')).toBeInTheDocument()
|
||||
})
|
||||
|
||||
it('displays config keys for each log', async () => {
|
||||
server.use(
|
||||
http.get('*/api/v1/config/sync-logs', () => {
|
||||
return HttpResponse.json(mockSyncLogs)
|
||||
}),
|
||||
)
|
||||
renderWithProviders(<ConfigSync />)
|
||||
|
||||
await waitFor(() => {
|
||||
expect(screen.getByText('model_config,prompt_config')).toBeInTheDocument()
|
||||
})
|
||||
})
|
||||
|
||||
it('displays resolution column', async () => {
|
||||
server.use(
|
||||
http.get('*/api/v1/config/sync-logs', () => {
|
||||
return HttpResponse.json(mockSyncLogs)
|
||||
}),
|
||||
)
|
||||
renderWithProviders(<ConfigSync />)
|
||||
|
||||
await waitFor(() => {
|
||||
expect(screen.getByText('client_wins')).toBeInTheDocument()
|
||||
})
|
||||
})
|
||||
|
||||
it('color-codes action tags', async () => {
|
||||
server.use(
|
||||
http.get('*/api/v1/config/sync-logs', () => {
|
||||
return HttpResponse.json(mockSyncLogs)
|
||||
}),
|
||||
)
|
||||
renderWithProviders(<ConfigSync />)
|
||||
|
||||
await waitFor(() => {
|
||||
const pushTag = screen.getByText('推送').closest('.ant-tag')
|
||||
expect(pushTag?.className).toMatch(/blue/)
|
||||
})
|
||||
const pullTag = screen.getByText('拉取').closest('.ant-tag')
|
||||
expect(pullTag?.className).toMatch(/cyan/)
|
||||
})
|
||||
|
||||
it('renders table column headers', async () => {
|
||||
server.use(
|
||||
http.get('*/api/v1/config/sync-logs', () => {
|
||||
return HttpResponse.json(mockSyncLogs)
|
||||
}),
|
||||
)
|
||||
renderWithProviders(<ConfigSync />)
|
||||
|
||||
await waitFor(() => {
|
||||
expect(screen.getByText('操作')).toBeInTheDocument()
|
||||
})
|
||||
expect(screen.getByText('客户端指纹')).toBeInTheDocument()
|
||||
expect(screen.getByText('配置键')).toBeInTheDocument()
|
||||
})
|
||||
})
|
||||
Reference in New Issue
Block a user