feat(L+M): configurable dashboard widget system + test framework
Phase L: Dashboard widget system - Migration 046: dashboard_configs table (user/tenant/role fallback cascade) - DashboardConfig model + dashboard_service with get/upsert per-user and tenant-default - API router: GET/PUT /api/dashboard/config, GET/PUT /api/dashboard/tenant-default - Frontend: 5 widget components (ProductionStats, QueueStatus, RecentRenders, CostOverview, WorkerStatus) - DashboardGrid with API-backed config, DashboardCustomizeModal (checkbox selection, save/cancel) - Dashboard.tsx: widget grid + analytics section (privileged users) - Admin.tsx: restructured with new section order and Maintenance 2-col grid Phase M: Test framework - Backend: pytest-asyncio + pytest-cov + factory-boy in pyproject.toml - conftest.py: excel_dir fixtures + async DB fixtures + mock storage/celery stubs - Domain tests: billing_service, cache_service, notifications_service, imports_sanity - Integration: test_api_health.py smoke test (requires running backend) - Frontend: vitest + @testing-library/react + msw added to package.json - vite.config.ts: test block (jsdom + globals + setupFiles) - tsconfig.json: exclude src/__tests__ from main tsc (test runner handles its own types) - MSW handlers for /api/auth/me, Billing.test.tsx, formatters.test.ts Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
@@ -0,0 +1,37 @@
|
||||
import { http, HttpResponse } from 'msw'
|
||||
|
||||
export const handlers = [
|
||||
http.get('/api/admin/settings', () => {
|
||||
return HttpResponse.json({
|
||||
blender_engine: 'cycles',
|
||||
blender_cycles_samples: 256,
|
||||
blender_eevee_samples: 64,
|
||||
thumbnail_format: 'jpg',
|
||||
stl_quality: 'low',
|
||||
blender_smooth_angle: 30,
|
||||
cycles_device: 'auto',
|
||||
blender_max_concurrent_renders: 3,
|
||||
render_stall_timeout_minutes: 120,
|
||||
render_backend: 'celery',
|
||||
product_thumbnail_priority: '["latest_render","cad_thumbnail"]',
|
||||
smtp_enabled: false,
|
||||
smtp_host: '',
|
||||
smtp_port: 587,
|
||||
smtp_user: '',
|
||||
smtp_password: '',
|
||||
smtp_from_address: '',
|
||||
})
|
||||
}),
|
||||
http.get('/api/billing/invoices', () => {
|
||||
return HttpResponse.json([])
|
||||
}),
|
||||
http.get('/api/notifications/config', () => {
|
||||
return HttpResponse.json([])
|
||||
}),
|
||||
http.get('/api/dashboard/config', () => {
|
||||
return HttpResponse.json([
|
||||
{ widget_type: 'ProductionStats', position: { col: 0, row: 0, w: 2, h: 1 } },
|
||||
{ widget_type: 'QueueStatus', position: { col: 2, row: 0, w: 1, h: 1 } },
|
||||
])
|
||||
}),
|
||||
]
|
||||
@@ -0,0 +1,3 @@
|
||||
import { setupServer } from 'msw/node'
|
||||
import { handlers } from './handlers'
|
||||
export const server = setupServer(...handlers)
|
||||
@@ -0,0 +1,9 @@
|
||||
import { describe, test, expect } from 'vitest'
|
||||
|
||||
// Minimaler Test: Billing-Seite kann importiert werden ohne Crash
|
||||
describe('Billing Page', () => {
|
||||
test('renders without crashing', async () => {
|
||||
const module = await import('../../pages/Billing')
|
||||
expect(module.default).toBeDefined()
|
||||
})
|
||||
})
|
||||
@@ -0,0 +1 @@
|
||||
import '@testing-library/jest-dom'
|
||||
@@ -0,0 +1,16 @@
|
||||
import { describe, test, expect } from 'vitest'
|
||||
|
||||
// Teste pure utility-Funktionen
|
||||
describe('Formatter utilities', () => {
|
||||
test('EUR formatting', () => {
|
||||
const amount = 1234.56
|
||||
const formatted = new Intl.NumberFormat('de-DE', { style: 'currency', currency: 'EUR' }).format(amount)
|
||||
expect(formatted).toContain('1.234,56')
|
||||
})
|
||||
|
||||
test('date formatting', () => {
|
||||
const d = new Date('2026-03-06T00:00:00Z')
|
||||
const iso = d.toISOString().slice(0, 10)
|
||||
expect(iso).toBe('2026-03-06')
|
||||
})
|
||||
})
|
||||
Reference in New Issue
Block a user