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:
2026-03-06 21:50:07 +01:00
parent 19c15adbee
commit bfc0050580
38 changed files with 4210 additions and 13 deletions
+37
View File
@@ -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 } },
])
}),
]
+3
View File
@@ -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()
})
})
+1
View File
@@ -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')
})
})