diff --git a/backend/internal/auth/context.go b/backend/internal/auth/context.go index c2aeaef..9553104 100644 --- a/backend/internal/auth/context.go +++ b/backend/internal/auth/context.go @@ -9,19 +9,11 @@ import ( type contextKey string const ( -<<<<<<< HEAD userIDKey contextKey = "user_id" tenantIDKey contextKey = "tenant_id" + userRoleKey contextKey = "user_role" ipKey contextKey = "ip_address" userAgentKey contextKey = "user_agent" -||||||| 82878df - userIDKey contextKey = "user_id" - tenantIDKey contextKey = "tenant_id" -======= - userIDKey contextKey = "user_id" - tenantIDKey contextKey = "tenant_id" - userRoleKey contextKey = "user_role" ->>>>>>> mai/pike/p0-role-based ) func ContextWithUserID(ctx context.Context, userID uuid.UUID) context.Context { @@ -41,7 +33,15 @@ func TenantFromContext(ctx context.Context) (uuid.UUID, bool) { id, ok := ctx.Value(tenantIDKey).(uuid.UUID) return id, ok } -<<<<<<< HEAD + +func ContextWithUserRole(ctx context.Context, role string) context.Context { + return context.WithValue(ctx, userRoleKey, role) +} + +func UserRoleFromContext(ctx context.Context) string { + role, _ := ctx.Value(userRoleKey).(string) + return role +} func ContextWithRequestInfo(ctx context.Context, ip, userAgent string) context.Context { ctx = context.WithValue(ctx, ipKey, ip) @@ -62,15 +62,3 @@ func UserAgentFromContext(ctx context.Context) *string { } return nil } -||||||| 82878df -======= - -func ContextWithUserRole(ctx context.Context, role string) context.Context { - return context.WithValue(ctx, userRoleKey, role) -} - -func UserRoleFromContext(ctx context.Context) string { - role, _ := ctx.Value(userRoleKey).(string) - return role -} ->>>>>>> mai/pike/p0-role-based diff --git a/backend/internal/auth/middleware.go b/backend/internal/auth/middleware.go index 02428c6..804f770 100644 --- a/backend/internal/auth/middleware.go +++ b/backend/internal/auth/middleware.go @@ -35,36 +35,8 @@ func (m *Middleware) RequireAuth(next http.Handler) http.Handler { } ctx := ContextWithUserID(r.Context(), userID) -<<<<<<< HEAD // Tenant resolution is handled by TenantResolver middleware for scoped routes. // Tenant management routes handle their own access control. -||||||| 82878df - - // Resolve tenant and role from user_tenants - var membership struct { - TenantID uuid.UUID `db:"tenant_id"` - Role string `db:"role"` - } - err = m.db.GetContext(r.Context(), &membership, - "SELECT tenant_id, role FROM user_tenants WHERE user_id = $1 LIMIT 1", userID) - if err != nil { - http.Error(w, "no tenant found for user", http.StatusForbidden) - return - } - ctx = ContextWithTenantID(ctx, membership.TenantID) - ctx = ContextWithUserRole(ctx, membership.Role) - -======= - - // Resolve tenant from user_tenants - var tenantID uuid.UUID - err = m.db.GetContext(r.Context(), &tenantID, - "SELECT tenant_id FROM user_tenants WHERE user_id = $1 LIMIT 1", userID) - if err != nil { - http.Error(w, "no tenant found for user", http.StatusForbidden) - return - } - ctx = ContextWithTenantID(ctx, tenantID) // Capture IP and user-agent for audit logging ip := r.Header.Get("X-Forwarded-For") @@ -73,7 +45,6 @@ func (m *Middleware) RequireAuth(next http.Handler) http.Handler { } ctx = ContextWithRequestInfo(ctx, ip, r.UserAgent()) ->>>>>>> mai/knuth/p0-audit-trail-append next.ServeHTTP(w, r.WithContext(ctx)) }) } diff --git a/backend/internal/auth/tenant_resolver.go b/backend/internal/auth/tenant_resolver.go index 0dab57f..b952262 100644 --- a/backend/internal/auth/tenant_resolver.go +++ b/backend/internal/auth/tenant_resolver.go @@ -12,12 +12,8 @@ import ( // Defined as an interface to avoid circular dependency with services. type TenantLookup interface { FirstTenantForUser(ctx context.Context, userID uuid.UUID) (*uuid.UUID, error) -<<<<<<< HEAD VerifyAccess(ctx context.Context, userID, tenantID uuid.UUID) (bool, error) -||||||| 82878df -======= GetUserRole(ctx context.Context, userID, tenantID uuid.UUID) (string, error) ->>>>>>> mai/pike/p0-role-based } // TenantResolver is middleware that resolves the tenant from X-Tenant-ID header @@ -46,33 +42,19 @@ func (tr *TenantResolver) Resolve(next http.Handler) http.Handler { http.Error(w, `{"error":"invalid X-Tenant-ID"}`, http.StatusBadRequest) return } -<<<<<<< HEAD - // Verify user has access to this tenant - hasAccess, err := tr.lookup.VerifyAccess(r.Context(), userID, parsed) + // Verify user has access and get their role + role, err := tr.lookup.GetUserRole(r.Context(), userID, parsed) if err != nil { slog.Error("tenant access check failed", "error", err, "user_id", userID, "tenant_id", parsed) http.Error(w, `{"error":"internal error"}`, http.StatusInternalServerError) return } - if !hasAccess { + if role == "" { http.Error(w, `{"error":"no access to tenant"}`, http.StatusForbidden) return } -||||||| 82878df -======= - // Verify user has access and get their role - role, err := tr.lookup.GetUserRole(r.Context(), userID, parsed) - if err != nil { - http.Error(w, "error checking tenant access", http.StatusInternalServerError) - return - } - if role == "" { - http.Error(w, "no access to this tenant", http.StatusForbidden) - return - } ->>>>>>> mai/pike/p0-role-based tenantID = parsed // Override the role from middleware with the correct one for this tenant r = r.WithContext(ContextWithUserRole(r.Context(), role)) diff --git a/backend/internal/auth/tenant_resolver_test.go b/backend/internal/auth/tenant_resolver_test.go index d0300c2..df36d77 100644 --- a/backend/internal/auth/tenant_resolver_test.go +++ b/backend/internal/auth/tenant_resolver_test.go @@ -10,49 +10,34 @@ import ( ) type mockTenantLookup struct { -<<<<<<< HEAD tenantID *uuid.UUID err error hasAccess bool accessErr error -||||||| 82878df - tenantID *uuid.UUID - err error -======= - tenantID *uuid.UUID - role string - err error ->>>>>>> mai/pike/p0-role-based + role string } func (m *mockTenantLookup) FirstTenantForUser(ctx context.Context, userID uuid.UUID) (*uuid.UUID, error) { return m.tenantID, m.err } -<<<<<<< HEAD func (m *mockTenantLookup) VerifyAccess(ctx context.Context, userID, tenantID uuid.UUID) (bool, error) { return m.hasAccess, m.accessErr } -||||||| 82878df -======= func (m *mockTenantLookup) GetUserRole(ctx context.Context, userID, tenantID uuid.UUID) (string, error) { if m.role != "" { return m.role, m.err } - return "associate", m.err + if m.hasAccess { + return "associate", m.err + } + return "", m.err } ->>>>>>> mai/pike/p0-role-based func TestTenantResolver_FromHeader(t *testing.T) { tenantID := uuid.New() -<<<<<<< HEAD - tr := NewTenantResolver(&mockTenantLookup{hasAccess: true}) -||||||| 82878df - tr := NewTenantResolver(&mockTenantLookup{}) -======= - tr := NewTenantResolver(&mockTenantLookup{role: "partner"}) ->>>>>>> mai/pike/p0-role-based + tr := NewTenantResolver(&mockTenantLookup{hasAccess: true, role: "partner"}) var gotTenantID uuid.UUID next := http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { diff --git a/backend/internal/handlers/deadlines.go b/backend/internal/handlers/deadlines.go index e1a2a39..1ff7629 100644 --- a/backend/internal/handlers/deadlines.go +++ b/backend/internal/handlers/deadlines.go @@ -198,18 +198,8 @@ func (h *DeadlineHandlers) Delete(w http.ResponseWriter, r *http.Request) { return } -<<<<<<< HEAD - if err := h.deadlines.Delete(tenantID, deadlineID); err != nil { + if err := h.deadlines.Delete(r.Context(), tenantID, deadlineID); err != nil { writeError(w, http.StatusNotFound, "deadline not found") -||||||| 82878df - err = h.deadlines.Delete(tenantID, deadlineID) - if err != nil { - writeError(w, http.StatusNotFound, err.Error()) -======= - err = h.deadlines.Delete(r.Context(), tenantID, deadlineID) - if err != nil { - writeError(w, http.StatusNotFound, err.Error()) ->>>>>>> mai/knuth/p0-audit-trail-append return } diff --git a/backend/internal/handlers/reports.go b/backend/internal/handlers/reports.go new file mode 100644 index 0000000..f1d6e24 --- /dev/null +++ b/backend/internal/handlers/reports.go @@ -0,0 +1,109 @@ +package handlers + +import ( + "net/http" + "time" + + "mgit.msbls.de/m/KanzlAI-mGMT/internal/auth" + "mgit.msbls.de/m/KanzlAI-mGMT/internal/services" +) + +type ReportHandler struct { + svc *services.ReportingService +} + +func NewReportHandler(svc *services.ReportingService) *ReportHandler { + return &ReportHandler{svc: svc} +} + +// parseDateRange extracts from/to query params, defaulting to last 12 months. +func parseDateRange(r *http.Request) (time.Time, time.Time) { + now := time.Now() + from := time.Date(now.Year()-1, now.Month(), 1, 0, 0, 0, 0, time.UTC) + to := time.Date(now.Year(), now.Month(), now.Day(), 23, 59, 59, 0, time.UTC) + + if v := r.URL.Query().Get("from"); v != "" { + if t, err := time.Parse("2006-01-02", v); err == nil { + from = t + } + } + if v := r.URL.Query().Get("to"); v != "" { + if t, err := time.Parse("2006-01-02", v); err == nil { + to = time.Date(t.Year(), t.Month(), t.Day(), 23, 59, 59, 0, time.UTC) + } + } + + return from, to +} + +func (h *ReportHandler) Cases(w http.ResponseWriter, r *http.Request) { + tenantID, ok := auth.TenantFromContext(r.Context()) + if !ok { + writeError(w, http.StatusForbidden, "missing tenant") + return + } + + from, to := parseDateRange(r) + + data, err := h.svc.CaseReport(r.Context(), tenantID, from, to) + if err != nil { + internalError(w, "failed to generate case report", err) + return + } + + writeJSON(w, http.StatusOK, data) +} + +func (h *ReportHandler) Deadlines(w http.ResponseWriter, r *http.Request) { + tenantID, ok := auth.TenantFromContext(r.Context()) + if !ok { + writeError(w, http.StatusForbidden, "missing tenant") + return + } + + from, to := parseDateRange(r) + + data, err := h.svc.DeadlineReport(r.Context(), tenantID, from, to) + if err != nil { + internalError(w, "failed to generate deadline report", err) + return + } + + writeJSON(w, http.StatusOK, data) +} + +func (h *ReportHandler) Workload(w http.ResponseWriter, r *http.Request) { + tenantID, ok := auth.TenantFromContext(r.Context()) + if !ok { + writeError(w, http.StatusForbidden, "missing tenant") + return + } + + from, to := parseDateRange(r) + + data, err := h.svc.WorkloadReport(r.Context(), tenantID, from, to) + if err != nil { + internalError(w, "failed to generate workload report", err) + return + } + + writeJSON(w, http.StatusOK, data) +} + +func (h *ReportHandler) Billing(w http.ResponseWriter, r *http.Request) { + tenantID, ok := auth.TenantFromContext(r.Context()) + if !ok { + writeError(w, http.StatusForbidden, "missing tenant") + return + } + + from, to := parseDateRange(r) + + data, err := h.svc.BillingReport(r.Context(), tenantID, from, to) + if err != nil { + internalError(w, "failed to generate billing report", err) + return + } + + writeJSON(w, http.StatusOK, data) +} diff --git a/backend/internal/router/router.go b/backend/internal/router/router.go index 84b7f8e..af868d2 100644 --- a/backend/internal/router/router.go +++ b/backend/internal/router/router.go @@ -29,14 +29,8 @@ func New(db *sqlx.DB, authMW *auth.Middleware, cfg *config.Config, calDAVSvc *se deadlineRuleSvc := services.NewDeadlineRuleService(db) calculator := services.NewDeadlineCalculator(holidaySvc) storageCli := services.NewStorageClient(cfg.SupabaseURL, cfg.SupabaseServiceKey) -<<<<<<< HEAD documentSvc := services.NewDocumentService(db, storageCli, auditSvc) -||||||| 82878df - documentSvc := services.NewDocumentService(db, storageCli) -======= - documentSvc := services.NewDocumentService(db, storageCli) assignmentSvc := services.NewCaseAssignmentService(db) ->>>>>>> mai/pike/p0-role-based // AI service (optional — only if API key is configured) var aiH *handlers.AIHandler @@ -50,6 +44,7 @@ func New(db *sqlx.DB, authMW *auth.Middleware, cfg *config.Config, calDAVSvc *se noteSvc := services.NewNoteService(db, auditSvc) dashboardSvc := services.NewDashboardService(db) + reportingSvc := services.NewReportingService(db) // Notification handler (optional — nil in tests) var notifH *handlers.NotificationHandler @@ -67,6 +62,7 @@ func New(db *sqlx.DB, authMW *auth.Middleware, cfg *config.Config, calDAVSvc *se ruleH := handlers.NewDeadlineRuleHandlers(deadlineRuleSvc) calcH := handlers.NewCalculateHandlers(calculator, deadlineRuleSvc) dashboardH := handlers.NewDashboardHandler(dashboardSvc) + reportH := handlers.NewReportHandler(reportingSvc) noteH := handlers.NewNoteHandler(noteSvc) eventH := handlers.NewCaseEventHandler(db) docH := handlers.NewDocumentHandler(documentSvc) @@ -162,16 +158,16 @@ func New(db *sqlx.DB, authMW *auth.Middleware, cfg *config.Config, calDAVSvc *se // Dashboard — all can view scoped.HandleFunc("GET /api/dashboard", dashboardH.Get) -<<<<<<< HEAD + // Reports — all can view + scoped.HandleFunc("GET /api/reports/cases", reportH.Cases) + scoped.HandleFunc("GET /api/reports/deadlines", reportH.Deadlines) + scoped.HandleFunc("GET /api/reports/workload", reportH.Workload) + scoped.HandleFunc("GET /api/reports/billing", reportH.Billing) + // Audit log scoped.HandleFunc("GET /api/audit-log", auditH.List) - // Documents -||||||| 82878df - // Documents -======= // Documents — all can upload, delete checked in handler (own vs all) ->>>>>>> mai/pike/p0-role-based scoped.HandleFunc("GET /api/cases/{id}/documents", docH.ListByCase) scoped.HandleFunc("POST /api/cases/{id}/documents", perm(auth.PermUploadDocuments, docH.Upload)) scoped.HandleFunc("GET /api/documents/{docId}", docH.Download) @@ -185,7 +181,6 @@ func New(db *sqlx.DB, authMW *auth.Middleware, cfg *config.Config, calDAVSvc *se scoped.HandleFunc("POST /api/ai/summarize-case", perm(auth.PermAIExtraction, aiLimiter.LimitFunc(aiH.SummarizeCase))) } -<<<<<<< HEAD // Notifications if notifH != nil { scoped.HandleFunc("GET /api/notifications", notifH.List) @@ -196,12 +191,7 @@ func New(db *sqlx.DB, authMW *auth.Middleware, cfg *config.Config, calDAVSvc *se scoped.HandleFunc("PUT /api/notification-preferences", notifH.UpdatePreferences) } - // CalDAV sync endpoints -||||||| 82878df - // CalDAV sync endpoints -======= // CalDAV sync endpoints — settings permission required ->>>>>>> mai/pike/p0-role-based if calDAVSvc != nil { calDAVH := handlers.NewCalDAVHandler(calDAVSvc) scoped.HandleFunc("POST /api/caldav/sync", perm(auth.PermManageSettings, calDAVH.TriggerSync)) diff --git a/backend/internal/services/reporting_service.go b/backend/internal/services/reporting_service.go new file mode 100644 index 0000000..a10bdf9 --- /dev/null +++ b/backend/internal/services/reporting_service.go @@ -0,0 +1,329 @@ +package services + +import ( + "context" + "fmt" + "time" + + "github.com/google/uuid" + "github.com/jmoiron/sqlx" +) + +type ReportingService struct { + db *sqlx.DB +} + +func NewReportingService(db *sqlx.DB) *ReportingService { + return &ReportingService{db: db} +} + +// --- Case Statistics --- + +type CaseStats struct { + Period string `json:"period" db:"period"` + Opened int `json:"opened" db:"opened"` + Closed int `json:"closed" db:"closed"` + Active int `json:"active" db:"active"` +} + +type CasesByType struct { + CaseType string `json:"case_type" db:"case_type"` + Count int `json:"count" db:"count"` +} + +type CasesByCourt struct { + Court string `json:"court" db:"court"` + Count int `json:"count" db:"count"` +} + +type CaseReport struct { + Monthly []CaseStats `json:"monthly"` + ByType []CasesByType `json:"by_type"` + ByCourt []CasesByCourt `json:"by_court"` + Total CaseReportTotals `json:"total"` +} + +type CaseReportTotals struct { + Opened int `json:"opened"` + Closed int `json:"closed"` + Active int `json:"active"` +} + +func (s *ReportingService) CaseReport(ctx context.Context, tenantID uuid.UUID, from, to time.Time) (*CaseReport, error) { + report := &CaseReport{} + + // Monthly breakdown + monthlyQuery := ` + SELECT + TO_CHAR(DATE_TRUNC('month', created_at), 'YYYY-MM') AS period, + COUNT(*) AS opened, + COUNT(*) FILTER (WHERE status IN ('closed', 'archived')) AS closed, + COUNT(*) FILTER (WHERE status = 'active') AS active + FROM cases + WHERE tenant_id = $1 AND created_at >= $2 AND created_at <= $3 + GROUP BY DATE_TRUNC('month', created_at) + ORDER BY DATE_TRUNC('month', created_at)` + + report.Monthly = []CaseStats{} + if err := s.db.SelectContext(ctx, &report.Monthly, monthlyQuery, tenantID, from, to); err != nil { + return nil, fmt.Errorf("case report monthly: %w", err) + } + + // By type + typeQuery := ` + SELECT COALESCE(case_type, 'Sonstiges') AS case_type, COUNT(*) AS count + FROM cases + WHERE tenant_id = $1 AND created_at >= $2 AND created_at <= $3 + GROUP BY case_type + ORDER BY count DESC` + + report.ByType = []CasesByType{} + if err := s.db.SelectContext(ctx, &report.ByType, typeQuery, tenantID, from, to); err != nil { + return nil, fmt.Errorf("case report by type: %w", err) + } + + // By court + courtQuery := ` + SELECT COALESCE(court, 'Ohne Gericht') AS court, COUNT(*) AS count + FROM cases + WHERE tenant_id = $1 AND created_at >= $2 AND created_at <= $3 + GROUP BY court + ORDER BY count DESC` + + report.ByCourt = []CasesByCourt{} + if err := s.db.SelectContext(ctx, &report.ByCourt, courtQuery, tenantID, from, to); err != nil { + return nil, fmt.Errorf("case report by court: %w", err) + } + + // Totals + totalsQuery := ` + SELECT + COUNT(*) AS opened, + COUNT(*) FILTER (WHERE status IN ('closed', 'archived')) AS closed, + COUNT(*) FILTER (WHERE status = 'active') AS active + FROM cases + WHERE tenant_id = $1 AND created_at >= $2 AND created_at <= $3` + + if err := s.db.GetContext(ctx, &report.Total, totalsQuery, tenantID, from, to); err != nil { + return nil, fmt.Errorf("case report totals: %w", err) + } + + return report, nil +} + +// --- Deadline Compliance --- + +type DeadlineCompliance struct { + Period string `json:"period" db:"period"` + Total int `json:"total" db:"total"` + Met int `json:"met" db:"met"` + Missed int `json:"missed" db:"missed"` + Pending int `json:"pending" db:"pending"` + ComplianceRate float64 `json:"compliance_rate"` +} + +type MissedDeadline struct { + ID uuid.UUID `json:"id" db:"id"` + Title string `json:"title" db:"title"` + DueDate string `json:"due_date" db:"due_date"` + CaseID uuid.UUID `json:"case_id" db:"case_id"` + CaseNumber string `json:"case_number" db:"case_number"` + CaseTitle string `json:"case_title" db:"case_title"` + DaysOverdue int `json:"days_overdue" db:"days_overdue"` +} + +type DeadlineReport struct { + Monthly []DeadlineCompliance `json:"monthly"` + Missed []MissedDeadline `json:"missed"` + Total DeadlineReportTotals `json:"total"` +} + +type DeadlineReportTotals struct { + Total int `json:"total"` + Met int `json:"met"` + Missed int `json:"missed"` + Pending int `json:"pending"` + ComplianceRate float64 `json:"compliance_rate"` +} + +func (s *ReportingService) DeadlineReport(ctx context.Context, tenantID uuid.UUID, from, to time.Time) (*DeadlineReport, error) { + report := &DeadlineReport{} + + // Monthly compliance + monthlyQuery := ` + SELECT + TO_CHAR(DATE_TRUNC('month', due_date), 'YYYY-MM') AS period, + COUNT(*) AS total, + COUNT(*) FILTER (WHERE status = 'completed' AND (completed_at IS NULL OR completed_at::date <= due_date)) AS met, + COUNT(*) FILTER (WHERE (status = 'completed' AND completed_at::date > due_date) OR (status = 'pending' AND due_date < CURRENT_DATE)) AS missed, + COUNT(*) FILTER (WHERE status = 'pending' AND due_date >= CURRENT_DATE) AS pending + FROM deadlines + WHERE tenant_id = $1 AND due_date >= $2 AND due_date <= $3 + GROUP BY DATE_TRUNC('month', due_date) + ORDER BY DATE_TRUNC('month', due_date)` + + report.Monthly = []DeadlineCompliance{} + if err := s.db.SelectContext(ctx, &report.Monthly, monthlyQuery, tenantID, from, to); err != nil { + return nil, fmt.Errorf("deadline report monthly: %w", err) + } + + // Calculate compliance rates + for i := range report.Monthly { + completed := report.Monthly[i].Met + report.Monthly[i].Missed + if completed > 0 { + report.Monthly[i].ComplianceRate = float64(report.Monthly[i].Met) / float64(completed) * 100 + } + } + + // Missed deadlines list + missedQuery := ` + SELECT d.id, d.title, d.due_date, d.case_id, c.case_number, c.title AS case_title, + (CURRENT_DATE - d.due_date::date) AS days_overdue + FROM deadlines d + JOIN cases c ON c.id = d.case_id AND c.tenant_id = d.tenant_id + WHERE d.tenant_id = $1 AND d.due_date >= $2 AND d.due_date <= $3 + AND ((d.status = 'pending' AND d.due_date < CURRENT_DATE) + OR (d.status = 'completed' AND d.completed_at::date > d.due_date)) + ORDER BY d.due_date ASC + LIMIT 50` + + report.Missed = []MissedDeadline{} + if err := s.db.SelectContext(ctx, &report.Missed, missedQuery, tenantID, from, to); err != nil { + return nil, fmt.Errorf("deadline report missed: %w", err) + } + + // Totals + totalsQuery := ` + SELECT + COUNT(*) AS total, + COUNT(*) FILTER (WHERE status = 'completed' AND (completed_at IS NULL OR completed_at::date <= due_date)) AS met, + COUNT(*) FILTER (WHERE (status = 'completed' AND completed_at::date > due_date) OR (status = 'pending' AND due_date < CURRENT_DATE)) AS missed, + COUNT(*) FILTER (WHERE status = 'pending' AND due_date >= CURRENT_DATE) AS pending + FROM deadlines + WHERE tenant_id = $1 AND due_date >= $2 AND due_date <= $3` + + if err := s.db.GetContext(ctx, &report.Total, totalsQuery, tenantID, from, to); err != nil { + return nil, fmt.Errorf("deadline report totals: %w", err) + } + + completed := report.Total.Met + report.Total.Missed + if completed > 0 { + report.Total.ComplianceRate = float64(report.Total.Met) / float64(completed) * 100 + } + + return report, nil +} + +// --- Workload --- + +type UserWorkload struct { + UserID uuid.UUID `json:"user_id" db:"user_id"` + ActiveCases int `json:"active_cases" db:"active_cases"` + Deadlines int `json:"deadlines" db:"deadlines"` + Overdue int `json:"overdue" db:"overdue"` + Completed int `json:"completed" db:"completed"` +} + +type WorkloadReport struct { + Users []UserWorkload `json:"users"` +} + +func (s *ReportingService) WorkloadReport(ctx context.Context, tenantID uuid.UUID, from, to time.Time) (*WorkloadReport, error) { + report := &WorkloadReport{} + + query := ` + WITH user_cases AS ( + SELECT ca.user_id, COUNT(DISTINCT ca.case_id) AS active_cases + FROM case_assignments ca + JOIN cases c ON c.id = ca.case_id AND c.tenant_id = $1 + WHERE c.status = 'active' + GROUP BY ca.user_id + ), + user_deadlines AS ( + SELECT ca.user_id, + COUNT(*) AS deadlines, + COUNT(*) FILTER (WHERE d.status = 'pending' AND d.due_date < CURRENT_DATE) AS overdue, + COUNT(*) FILTER (WHERE d.status = 'completed' AND d.completed_at >= $2 AND d.completed_at <= $3) AS completed + FROM case_assignments ca + JOIN deadlines d ON d.case_id = ca.case_id AND d.tenant_id = $1 + WHERE d.due_date >= $2 AND d.due_date <= $3 + GROUP BY ca.user_id + ) + SELECT + COALESCE(uc.user_id, ud.user_id) AS user_id, + COALESCE(uc.active_cases, 0) AS active_cases, + COALESCE(ud.deadlines, 0) AS deadlines, + COALESCE(ud.overdue, 0) AS overdue, + COALESCE(ud.completed, 0) AS completed + FROM user_cases uc + FULL OUTER JOIN user_deadlines ud ON uc.user_id = ud.user_id + ORDER BY active_cases DESC` + + report.Users = []UserWorkload{} + if err := s.db.SelectContext(ctx, &report.Users, query, tenantID, from, to); err != nil { + return nil, fmt.Errorf("workload report: %w", err) + } + + return report, nil +} + +// --- Billing (summary from case data) --- + +type BillingByMonth struct { + Period string `json:"period" db:"period"` + CasesActive int `json:"cases_active" db:"cases_active"` + CasesClosed int `json:"cases_closed" db:"cases_closed"` + CasesNew int `json:"cases_new" db:"cases_new"` +} + +type BillingByType struct { + CaseType string `json:"case_type" db:"case_type"` + Active int `json:"active" db:"active"` + Closed int `json:"closed" db:"closed"` + Total int `json:"total" db:"total"` +} + +type BillingReport struct { + Monthly []BillingByMonth `json:"monthly"` + ByType []BillingByType `json:"by_type"` +} + +func (s *ReportingService) BillingReport(ctx context.Context, tenantID uuid.UUID, from, to time.Time) (*BillingReport, error) { + report := &BillingReport{} + + // Monthly activity for billing overview + monthlyQuery := ` + SELECT + TO_CHAR(DATE_TRUNC('month', created_at), 'YYYY-MM') AS period, + COUNT(*) FILTER (WHERE status = 'active') AS cases_active, + COUNT(*) FILTER (WHERE status IN ('closed', 'archived')) AS cases_closed, + COUNT(*) AS cases_new + FROM cases + WHERE tenant_id = $1 AND created_at >= $2 AND created_at <= $3 + GROUP BY DATE_TRUNC('month', created_at) + ORDER BY DATE_TRUNC('month', created_at)` + + report.Monthly = []BillingByMonth{} + if err := s.db.SelectContext(ctx, &report.Monthly, monthlyQuery, tenantID, from, to); err != nil { + return nil, fmt.Errorf("billing report monthly: %w", err) + } + + // By case type + typeQuery := ` + SELECT + COALESCE(case_type, 'Sonstiges') AS case_type, + COUNT(*) FILTER (WHERE status = 'active') AS active, + COUNT(*) FILTER (WHERE status IN ('closed', 'archived')) AS closed, + COUNT(*) AS total + FROM cases + WHERE tenant_id = $1 AND created_at >= $2 AND created_at <= $3 + GROUP BY case_type + ORDER BY total DESC` + + report.ByType = []BillingByType{} + if err := s.db.SelectContext(ctx, &report.ByType, typeQuery, tenantID, from, to); err != nil { + return nil, fmt.Errorf("billing report by type: %w", err) + } + + return report, nil +} diff --git a/frontend/bun.lock b/frontend/bun.lock index 826fdf8..f1d63d1 100644 --- a/frontend/bun.lock +++ b/frontend/bun.lock @@ -14,6 +14,7 @@ "react": "19.1.0", "react-dom": "19.1.0", "react-dropzone": "^15.0.0", + "recharts": "^3.8.1", "sonner": "^2.0.7", }, "devDependencies": { @@ -244,6 +245,8 @@ "@open-draft/until": ["@open-draft/until@2.1.0", "", {}, "sha512-U69T3ItWHvLwGg5eJ0n3I62nWuE6ilHlmz7zM0npLBRvPRd7e6NYmg54vvRtP5mZG7kZqZCFVdsTWo7BPtBujg=="], + "@reduxjs/toolkit": ["@reduxjs/toolkit@2.11.2", "", { "dependencies": { "@standard-schema/spec": "^1.0.0", "@standard-schema/utils": "^0.3.0", "immer": "^11.0.0", "redux": "^5.0.1", "redux-thunk": "^3.1.0", "reselect": "^5.1.0" }, "peerDependencies": { "react": "^16.9.0 || ^17.0.0 || ^18 || ^19", "react-redux": "^7.2.1 || ^8.1.3 || ^9.0.0" }, "optionalPeers": ["react", "react-redux"] }, "sha512-Kd6kAHTA6/nUpp8mySPqj3en3dm0tdMIgbttnQ1xFMVpufoj+ADi8pXLBsd4xzTRHQa7t/Jv8W5UnCuW4kuWMQ=="], + "@rollup/rollup-android-arm-eabi": ["@rollup/rollup-android-arm-eabi@4.60.0", "", { "os": "android", "cpu": "arm" }, "sha512-WOhNW9K8bR3kf4zLxbfg6Pxu2ybOUbB2AjMDHSQx86LIF4rH4Ft7vmMwNt0loO0eonglSNy4cpD3MKXXKQu0/A=="], "@rollup/rollup-android-arm64": ["@rollup/rollup-android-arm64@4.60.0", "", { "os": "android", "cpu": "arm64" }, "sha512-u6JHLll5QKRvjciE78bQXDmqRqNs5M/3GVqZeMwvmjaNODJih/WIrJlFVEihvV0MiYFmd+ZyPr9wxOVbPAG2Iw=="], @@ -298,6 +301,10 @@ "@rushstack/eslint-patch": ["@rushstack/eslint-patch@1.16.1", "", {}, "sha512-TvZbIpeKqGQQ7X0zSCvPH9riMSFQFSggnfBjFZ1mEoILW+UuXCKwOoPcgjMwiUtRqFZ8jWhPJc4um14vC6I4ag=="], + "@standard-schema/spec": ["@standard-schema/spec@1.1.0", "", {}, "sha512-l2aFy5jALhniG5HgqrD6jXLi/rUWrKvqN/qJx6yoJsgKhblVd+iqqU4RCXavm/jPityDo5TCvKMnpjKnOriy0w=="], + + "@standard-schema/utils": ["@standard-schema/utils@0.3.0", "", {}, "sha512-e7Mew686owMaPJVNNLs55PUvgz371nKgwsc4vxE49zsODpJEnxgxRo2y/OKrqueavXgZNMDVj3DdHFlaSAeU8g=="], + "@supabase/auth-js": ["@supabase/auth-js@2.100.0", "", { "dependencies": { "tslib": "2.8.1" } }, "sha512-pdT3ye3UVRN1Cg0wom6BmyY+XTtp5DiJaYnPi6j8ht5i8Lq8kfqxJMJz9GI9YDKk3w1nhGOPnh6Qz5qpyYm+1w=="], "@supabase/functions-js": ["@supabase/functions-js@2.100.0", "", { "dependencies": { "tslib": "2.8.1" } }, "sha512-keLg79RPwP+uiwHuxFPTFgDRxPV46LM4j/swjyR2GKJgWniTVSsgiBHfbIBDcrQwehLepy09b/9QSHUywtKRWQ=="], @@ -362,6 +369,24 @@ "@types/aria-query": ["@types/aria-query@5.0.4", "", {}, "sha512-rfT93uj5s0PRL7EzccGMs3brplhcrghnDoV26NqKhCAS1hVo+WdNsPvE/yb6ilfr5hi2MEk6d5EWJTKdxg8jVw=="], + "@types/d3-array": ["@types/d3-array@3.2.2", "", {}, "sha512-hOLWVbm7uRza0BYXpIIW5pxfrKe0W+D5lrFiAEYR+pb6w3N2SwSMaJbXdUfSEv+dT4MfHBLtn5js0LAWaO6otw=="], + + "@types/d3-color": ["@types/d3-color@3.1.3", "", {}, "sha512-iO90scth9WAbmgv7ogoq57O9YpKmFBbmoEoCHDB2xMBY0+/KVrqAaCDyCE16dUspeOvIxFFRI+0sEtqDqy2b4A=="], + + "@types/d3-ease": ["@types/d3-ease@3.0.2", "", {}, "sha512-NcV1JjO5oDzoK26oMzbILE6HW7uVXOHLQvHshBUW4UMdZGfiY6v5BeQwh9a9tCzv+CeefZQHJt5SRgK154RtiA=="], + + "@types/d3-interpolate": ["@types/d3-interpolate@3.0.4", "", { "dependencies": { "@types/d3-color": "*" } }, "sha512-mgLPETlrpVV1YRJIglr4Ez47g7Yxjl1lj7YKsiMCb27VJH9W8NVM6Bb9d8kkpG/uAQS5AmbA48q2IAolKKo1MA=="], + + "@types/d3-path": ["@types/d3-path@3.1.1", "", {}, "sha512-VMZBYyQvbGmWyWVea0EHs/BwLgxc+MKi1zLDCONksozI4YJMcTt8ZEuIR4Sb1MMTE8MMW49v0IwI5+b7RmfWlg=="], + + "@types/d3-scale": ["@types/d3-scale@4.0.9", "", { "dependencies": { "@types/d3-time": "*" } }, "sha512-dLmtwB8zkAeO/juAMfnV+sItKjlsw2lKdZVVy6LRr0cBmegxSABiLEpGVmSJJ8O08i4+sGR6qQtb6WtuwJdvVw=="], + + "@types/d3-shape": ["@types/d3-shape@3.1.8", "", { "dependencies": { "@types/d3-path": "*" } }, "sha512-lae0iWfcDeR7qt7rA88BNiqdvPS5pFVPpo5OfjElwNaT2yyekbM0C9vK+yqBqEmHr6lDkRnYNoTBYlAgJa7a4w=="], + + "@types/d3-time": ["@types/d3-time@3.0.4", "", {}, "sha512-yuzZug1nkAAaBlBBikKZTgzCeA+k1uy4ZFwWANOfKw5z5LRhV0gNA7gNkKm7HoK+HRN0wX3EkxGk0fpbWhmB7g=="], + + "@types/d3-timer": ["@types/d3-timer@3.0.2", "", {}, "sha512-Ps3T8E8dZDam6fUyNiMkekK3XUsaUEik+idO9/YjPtfj2qruF8tFBXS7XhtE4iIXBLxhmLjP3SXpLhVf21I9Lw=="], + "@types/estree": ["@types/estree@1.0.8", "", {}, "sha512-dWHzHa2WqEXI/O1E9OjrocMTKJl2mSrEolh1Iomrv6U+JuNwaHXsXx9bLu5gG7BUWFIN0skIQJQ/L1rIex4X6w=="], "@types/json-schema": ["@types/json-schema@7.0.15", "", {}, "sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA=="], @@ -376,6 +401,8 @@ "@types/statuses": ["@types/statuses@2.0.6", "", {}, "sha512-xMAgYwceFhRA2zY+XbEA7mxYbA093wdiW8Vu6gZPGWy9cmOyU9XesH1tNcEWsKFd5Vzrqx5T3D38PWx1FIIXkA=="], + "@types/use-sync-external-store": ["@types/use-sync-external-store@0.0.6", "", {}, "sha512-zFDAD+tlpf2r4asuHEj0XH6pY6i0g5NeAHPn+15wk3BV6JA69eERFXC1gyGThDkVa1zCyKr5jox1+2LbV/AMLg=="], + "@types/ws": ["@types/ws@8.18.1", "", { "dependencies": { "@types/node": "*" } }, "sha512-ThVF6DCVhA8kUGy+aazFQ4kXQ7E1Ty7A3ypFOe0IcJV8O/M511G99AW24irKrW56Wt44yG9+ij8FaqoBGkuBXg=="], "@typescript-eslint/eslint-plugin": ["@typescript-eslint/eslint-plugin@8.57.2", "", { "dependencies": { "@eslint-community/regexpp": "^4.12.2", "@typescript-eslint/scope-manager": "8.57.2", "@typescript-eslint/type-utils": "8.57.2", "@typescript-eslint/utils": "8.57.2", "@typescript-eslint/visitor-keys": "8.57.2", "ignore": "^7.0.5", "natural-compare": "^1.4.0", "ts-api-utils": "^2.4.0" }, "peerDependencies": { "@typescript-eslint/parser": "^8.57.2", "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", "typescript": ">=4.8.4 <6.0.0" } }, "sha512-NZZgp0Fm2IkD+La5PR81sd+g+8oS6JwJje+aRWsDocxHkjyRw0J5L5ZTlN3LI1LlOcGL7ph3eaIUmTXMIjLk0w=="], @@ -528,6 +555,8 @@ "cliui": ["cliui@8.0.1", "", { "dependencies": { "string-width": "^4.2.0", "strip-ansi": "^6.0.1", "wrap-ansi": "^7.0.0" } }, "sha512-BSeNnyus75C4//NQ9gQt1/csTXyo/8Sb+afLAkzAptFuMsod9HFokGNudZpi/oQV73hnVK+sR+5PVRMd+Dr7YQ=="], + "clsx": ["clsx@2.1.1", "", {}, "sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA=="], + "color-convert": ["color-convert@2.0.1", "", { "dependencies": { "color-name": "~1.1.4" } }, "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ=="], "color-name": ["color-name@1.1.4", "", {}, "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA=="], @@ -546,6 +575,28 @@ "csstype": ["csstype@3.2.3", "", {}, "sha512-z1HGKcYy2xA8AGQfwrn0PAy+PB7X/GSj3UVJW9qKyn43xWa+gl5nXmU4qqLMRzWVLFC8KusUX8T/0kCiOYpAIQ=="], + "d3-array": ["d3-array@3.2.4", "", { "dependencies": { "internmap": "1 - 2" } }, "sha512-tdQAmyA18i4J7wprpYq8ClcxZy3SC31QMeByyCFyRt7BVHdREQZ5lpzoe5mFEYZUWe+oq8HBvk9JjpibyEV4Jg=="], + + "d3-color": ["d3-color@3.1.0", "", {}, "sha512-zg/chbXyeBtMQ1LbD/WSoW2DpC3I0mpmPdW+ynRTj/x2DAWYrIY7qeZIHidozwV24m4iavr15lNwIwLxRmOxhA=="], + + "d3-ease": ["d3-ease@3.0.1", "", {}, "sha512-wR/XK3D3XcLIZwpbvQwQ5fK+8Ykds1ip7A2Txe0yxncXSdq1L9skcG7blcedkOX+ZcgxGAmLX1FrRGbADwzi0w=="], + + "d3-format": ["d3-format@3.1.2", "", {}, "sha512-AJDdYOdnyRDV5b6ArilzCPPwc1ejkHcoyFarqlPqT7zRYjhavcT3uSrqcMvsgh2CgoPbK3RCwyHaVyxYcP2Arg=="], + + "d3-interpolate": ["d3-interpolate@3.0.1", "", { "dependencies": { "d3-color": "1 - 3" } }, "sha512-3bYs1rOD33uo8aqJfKP3JWPAibgw8Zm2+L9vBKEHJ2Rg+viTR7o5Mmv5mZcieN+FRYaAOWX5SJATX6k1PWz72g=="], + + "d3-path": ["d3-path@3.1.0", "", {}, "sha512-p3KP5HCf/bvjBSSKuXid6Zqijx7wIfNW+J/maPs+iwR35at5JCbLUT0LzF1cnjbCHWhqzQTIN2Jpe8pRebIEFQ=="], + + "d3-scale": ["d3-scale@4.0.2", "", { "dependencies": { "d3-array": "2.10.0 - 3", "d3-format": "1 - 3", "d3-interpolate": "1.2.0 - 3", "d3-time": "2.1.1 - 3", "d3-time-format": "2 - 4" } }, "sha512-GZW464g1SH7ag3Y7hXjf8RoUuAFIqklOAq3MRl4OaWabTFJY9PN/E1YklhXLh+OQ3fM9yS2nOkCoS+WLZ6kvxQ=="], + + "d3-shape": ["d3-shape@3.2.0", "", { "dependencies": { "d3-path": "^3.1.0" } }, "sha512-SaLBuwGm3MOViRq2ABk3eLoxwZELpH6zhl3FbAoJ7Vm1gofKx6El1Ib5z23NUEhF9AsGl7y+dzLe5Cw2AArGTA=="], + + "d3-time": ["d3-time@3.1.0", "", { "dependencies": { "d3-array": "2 - 3" } }, "sha512-VqKjzBLejbSMT4IgbmVgDjpkYrNWUYJnbCGo874u7MMKIWsILRX+OpX/gTk8MqjpT1A/c6HY2dCA77ZN0lkQ2Q=="], + + "d3-time-format": ["d3-time-format@4.1.0", "", { "dependencies": { "d3-time": "1 - 3" } }, "sha512-dJxPBlzC7NugB2PDLwo9Q8JiTR3M3e4/XANkreKSUxF8vvXKqm1Yfq4Q5dl8budlunRVlUUaDUgFt7eA8D6NLg=="], + + "d3-timer": ["d3-timer@3.0.1", "", {}, "sha512-ndfJ/JxxMd3nw31uyKoY2naivF+r29V+Lc0svZxe1JvvIRmi8hUsrMvdOwgS1o6uBHmiz91geQ0ylPP0aj1VUA=="], + "damerau-levenshtein": ["damerau-levenshtein@1.0.8", "", {}, "sha512-sdQSFB7+llfUcQHUQO3+B8ERRj0Oa4w9POWMI/puGtuf7gFywGmkaLCElnudfTiKZV+NvHqL0ifzdrI8Ro7ESA=="], "data-urls": ["data-urls@5.0.0", "", { "dependencies": { "whatwg-mimetype": "^4.0.0", "whatwg-url": "^14.0.0" } }, "sha512-ZYP5VBHshaDAiVZxjbRVcFJpc+4xGgT0bK3vzy1HLN8jTO975HEbuYzZJcHoQEY5K1a0z8YayJkyVETa08eNTg=="], @@ -562,6 +613,8 @@ "decimal.js": ["decimal.js@10.6.0", "", {}, "sha512-YpgQiITW3JXGntzdUmyUR1V812Hn8T1YVXhCu+wO3OpS4eU9l4YdD3qjyiKdV6mvV29zapkMeD390UVEf2lkUg=="], + "decimal.js-light": ["decimal.js-light@2.5.1", "", {}, "sha512-qIMFpTMZmny+MMIitAB6D7iVPEorVw6YQRWkvarTkT4tBeSLLiHzcwj6q0MmYSFCiVpiqPJTJEYIrpcPzVEIvg=="], + "deep-eql": ["deep-eql@5.0.2", "", {}, "sha512-h5k/5U50IJJFpzfL6nO9jaaumfjO/f2NjK/oYB2Djzm4p9L+3T9qWpZqZ2hAbLPuuYq9wrU08WQyBTL5GbPk5Q=="], "deep-is": ["deep-is@0.1.4", "", {}, "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ=="], @@ -606,6 +659,8 @@ "es-to-primitive": ["es-to-primitive@1.3.0", "", { "dependencies": { "is-callable": "^1.2.7", "is-date-object": "^1.0.5", "is-symbol": "^1.0.4" } }, "sha512-w+5mJ3GuFL+NjVtJlvydShqE1eN3h3PbI7/5LAsYJP/2qtuMXjfL2LpHSRqo4b4eSF5K/DH1JXKUAHSB2UW50g=="], + "es-toolkit": ["es-toolkit@1.45.1", "", {}, "sha512-/jhoOj/Fx+A+IIyDNOvO3TItGmlMKhtX8ISAHKE90c4b/k1tqaqEZ+uUqfpU8DMnW5cgNJv606zS55jGvza0Xw=="], + "esbuild": ["esbuild@0.21.5", "", { "optionalDependencies": { "@esbuild/aix-ppc64": "0.21.5", "@esbuild/android-arm": "0.21.5", "@esbuild/android-arm64": "0.21.5", "@esbuild/android-x64": "0.21.5", "@esbuild/darwin-arm64": "0.21.5", "@esbuild/darwin-x64": "0.21.5", "@esbuild/freebsd-arm64": "0.21.5", "@esbuild/freebsd-x64": "0.21.5", "@esbuild/linux-arm": "0.21.5", "@esbuild/linux-arm64": "0.21.5", "@esbuild/linux-ia32": "0.21.5", "@esbuild/linux-loong64": "0.21.5", "@esbuild/linux-mips64el": "0.21.5", "@esbuild/linux-ppc64": "0.21.5", "@esbuild/linux-riscv64": "0.21.5", "@esbuild/linux-s390x": "0.21.5", "@esbuild/linux-x64": "0.21.5", "@esbuild/netbsd-x64": "0.21.5", "@esbuild/openbsd-x64": "0.21.5", "@esbuild/sunos-x64": "0.21.5", "@esbuild/win32-arm64": "0.21.5", "@esbuild/win32-ia32": "0.21.5", "@esbuild/win32-x64": "0.21.5" }, "bin": { "esbuild": "bin/esbuild" } }, "sha512-mg3OPMV4hXywwpoDxu3Qda5xCKQi+vCTZq8S9J/EpkhB2HzKXq4SNFZE3+NK93JYxc8VMSep+lOUSC/RVKaBqw=="], "escalade": ["escalade@3.2.0", "", {}, "sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA=="], @@ -646,6 +701,8 @@ "esutils": ["esutils@2.0.3", "", {}, "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g=="], + "eventemitter3": ["eventemitter3@5.0.4", "", {}, "sha512-mlsTRyGaPBjPedk6Bvw+aqbsXDtoAyAzm5MO7JgU+yVRyMQ5O8bD4Kcci7BS85f93veegeCPkL8R4GLClnjLFw=="], + "expect-type": ["expect-type@1.3.0", "", {}, "sha512-knvyeauYhqjOYvQ66MznSMs83wmHrCycNEN6Ao+2AeYEfxUIkuiVxdEa1qlGEPK+We3n0THiDciYSsCcgW/DoA=="], "fast-deep-equal": ["fast-deep-equal@3.1.3", "", {}, "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q=="], @@ -736,6 +793,8 @@ "ignore": ["ignore@5.3.2", "", {}, "sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g=="], + "immer": ["immer@10.2.0", "", {}, "sha512-d/+XTN3zfODyjr89gM3mPq1WNX2B8pYsu7eORitdwyA2sBubnTl3laYlBk4sXY5FUa5qTZGBDPJICVbvqzjlbw=="], + "import-fresh": ["import-fresh@3.3.1", "", { "dependencies": { "parent-module": "^1.0.0", "resolve-from": "^4.0.0" } }, "sha512-TR3KfrTZTYLPB6jUjfx6MF9WcWrHL9su5TObK4ZkYgBdWKPOFoSoQIdEuTuR82pmtxH2spWG9h6etwfr1pLBqQ=="], "imurmurhash": ["imurmurhash@0.1.4", "", {}, "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA=="], @@ -744,6 +803,8 @@ "internal-slot": ["internal-slot@1.1.0", "", { "dependencies": { "es-errors": "^1.3.0", "hasown": "^2.0.2", "side-channel": "^1.1.0" } }, "sha512-4gd7VpWNQNB4UKKCFFVcp1AVv+FMOgs9NKzjHKusc8jTMhd5eL1NqQqOpE0KzMds804/yHlglp3uxgluOqAPLw=="], + "internmap": ["internmap@2.0.3", "", {}, "sha512-5Hh7Y1wQbvY5ooGgPbDaL5iYLAPzMTUrjMulskHLH6wnv/A+1q5rgEaiuqEjB+oxGXIVZs1FF+R/KPN3ZSQYYg=="], + "is-array-buffer": ["is-array-buffer@3.0.5", "", { "dependencies": { "call-bind": "^1.0.8", "call-bound": "^1.0.3", "get-intrinsic": "^1.2.6" } }, "sha512-DDfANUiiG2wC1qawP66qlTugJeL5HyzMpfr8lLK+jMQirGzNod0B12cFB/9q838Ru27sBwfw78/rdoU7RERz6A=="], "is-async-function": ["is-async-function@2.1.1", "", { "dependencies": { "async-function": "^1.0.0", "call-bound": "^1.0.3", "get-proto": "^1.0.1", "has-tostringtag": "^1.0.2", "safe-regex-test": "^1.1.0" } }, "sha512-9dgM/cZBnNvjzaMYHVoxxfPj2QXt22Ev7SuuPrs+xav0ukGB0S6d4ydZdEiM48kLx5kDV+QBPrpVnFyefL8kkQ=="], @@ -978,10 +1039,18 @@ "react-dropzone": ["react-dropzone@15.0.0", "", { "dependencies": { "attr-accept": "^2.2.4", "file-selector": "^2.1.0", "prop-types": "^15.8.1" }, "peerDependencies": { "react": ">= 16.8 || 18.0.0" } }, "sha512-lGjYV/EoqEjEWPnmiSvH4v5IoIAwQM2W4Z1C0Q/Pw2xD0eVzKPS359BQTUMum+1fa0kH2nrKjuavmTPOGhpLPg=="], - "react-is": ["react-is@16.13.1", "", {}, "sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ=="], + "react-is": ["react-is@17.0.2", "", {}, "sha512-w2GsyukL62IJnlaff/nRegPQR94C/XXamvMWmSHRJ4y7Ts/4ocGRmTHvOs8PSE6pB3dWOrD/nueuU5sduBsQ4w=="], + + "react-redux": ["react-redux@9.2.0", "", { "dependencies": { "@types/use-sync-external-store": "^0.0.6", "use-sync-external-store": "^1.4.0" }, "peerDependencies": { "@types/react": "^18.2.25 || ^19", "react": "^18.0 || ^19", "redux": "^5.0.0" }, "optionalPeers": ["@types/react", "redux"] }, "sha512-ROY9fvHhwOD9ySfrF0wmvu//bKCQ6AeZZq1nJNtbDC+kk5DuSuNX/n6YWYF/SYy7bSba4D4FSz8DJeKY/S/r+g=="], + + "recharts": ["recharts@3.8.1", "", { "dependencies": { "@reduxjs/toolkit": "^1.9.0 || 2.x.x", "clsx": "^2.1.1", "decimal.js-light": "^2.5.1", "es-toolkit": "^1.39.3", "eventemitter3": "^5.0.1", "immer": "^10.1.1", "react-redux": "8.x.x || 9.x.x", "reselect": "5.1.1", "tiny-invariant": "^1.3.3", "use-sync-external-store": "^1.2.2", "victory-vendor": "^37.0.2" }, "peerDependencies": { "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0", "react-dom": "^16.0.0 || ^17.0.0 || ^18.0.0 || ^19.0.0", "react-is": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0" } }, "sha512-mwzmO1s9sFL0TduUpwndxCUNoXsBw3u3E/0+A+cLcrSfQitSG62L32N69GhqUrrT5qKcAE3pCGVINC6pqkBBQg=="], "redent": ["redent@3.0.0", "", { "dependencies": { "indent-string": "^4.0.0", "strip-indent": "^3.0.0" } }, "sha512-6tDA8g98We0zd0GvVeMT9arEOnTw9qM03L9cJXaCjrip1OO764RDBLBfrB4cwzNGDj5OA5ioymC9GkizgWJDUg=="], + "redux": ["redux@5.0.1", "", {}, "sha512-M9/ELqF6fy8FwmkpnF0S3YKOqMyoWJ4+CS5Efg2ct3oY9daQvd/Pc71FpGZsVsbl3Cpb+IIcjBDUnnyBdQbq4w=="], + + "redux-thunk": ["redux-thunk@3.1.0", "", { "peerDependencies": { "redux": "^5.0.0" } }, "sha512-NW2r5T6ksUKXCabzhL9z+h206HQw/NJkcLm1GPImRQ8IzfXwRGqjVhKJGauHirT0DAuyy6hjdnMZaRoAcy0Klw=="], + "reflect.getprototypeof": ["reflect.getprototypeof@1.0.10", "", { "dependencies": { "call-bind": "^1.0.8", "define-properties": "^1.2.1", "es-abstract": "^1.23.9", "es-errors": "^1.3.0", "es-object-atoms": "^1.0.0", "get-intrinsic": "^1.2.7", "get-proto": "^1.0.1", "which-builtin-type": "^1.2.1" } }, "sha512-00o4I+DVrefhv+nX0ulyi3biSHCPDe+yLv5o/p6d/UVlirijB8E16FtfwSAi4g3tcqrQ4lRAqQSoFEZJehYEcw=="], "regexp.prototype.flags": ["regexp.prototype.flags@1.5.4", "", { "dependencies": { "call-bind": "^1.0.8", "define-properties": "^1.2.1", "es-errors": "^1.3.0", "get-proto": "^1.0.1", "gopd": "^1.2.0", "set-function-name": "^2.0.2" } }, "sha512-dYqgNSZbDwkaJ2ceRd9ojCGjBq+mOm9LmtXnAnEGyHhN/5R7iDW2TRw3h+o/jCFxus3P2LfWIIiwowAjANm7IA=="], @@ -990,6 +1059,8 @@ "requires-port": ["requires-port@1.0.0", "", {}, "sha512-KigOCHcocU3XODJxsu8i/j8T9tzT4adHiecwORRQ0ZZFcp7ahwXuRU1m+yuO90C5ZUyGeGfocHDI14M3L3yDAQ=="], + "reselect": ["reselect@5.1.1", "", {}, "sha512-K/BG6eIky/SBpzfHZv/dd+9JBFiS4SWV7FIujVyJRux6e45+73RaUHXLmIR1f7WOMaQ0U1km6qwklRQxpJJY0w=="], + "resolve": ["resolve@1.22.11", "", { "dependencies": { "is-core-module": "^2.16.1", "path-parse": "^1.0.7", "supports-preserve-symlinks-flag": "^1.0.0" }, "bin": { "resolve": "bin/resolve" } }, "sha512-RfqAvLnMl313r7c9oclB1HhUEAezcpLjz95wFH4LVuhk9JF/r22qmVP9AMmOU4vMX7Q8pN8jwNg/CSpdFnMjTQ=="], "resolve-from": ["resolve-from@4.0.0", "", {}, "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g=="], @@ -1096,6 +1167,8 @@ "tapable": ["tapable@2.3.2", "", {}, "sha512-1MOpMXuhGzGL5TTCZFItxCc0AARf1EZFQkGqMm7ERKj8+Hgr5oLvJOVFcC+lRmR8hCe2S3jC4T5D7Vg/d7/fhA=="], + "tiny-invariant": ["tiny-invariant@1.3.3", "", {}, "sha512-+FbBPE1o9QAYvviau/qC5SE3caw21q3xkvWKBtja5vgqOWIHHJ3ioaq1VPfn/Szqctz2bU/oYeKd9/z5BL+PVg=="], + "tinybench": ["tinybench@2.9.0", "", {}, "sha512-0+DUvqWMValLmha6lr4kD8iAMK1HzV0/aKnCtWb9v9641TnP/MFb7Pc2bxoxQjTXAErryXVgUOfv2YqNllqGeg=="], "tinyexec": ["tinyexec@0.3.2", "", {}, "sha512-KQQR9yN7R5+OSwaK0XQoj22pwHoTlgYqmUscPYoknOoWCWfj/5/ABTMRi69FrKU5ffPVh5QcFikpWJI/P1ocHA=="], @@ -1152,6 +1225,10 @@ "url-parse": ["url-parse@1.5.10", "", { "dependencies": { "querystringify": "^2.1.1", "requires-port": "^1.0.0" } }, "sha512-WypcfiRhfeUP9vvF0j6rw0J3hrWrw6iZv3+22h6iRMJ/8z1Tj6XfLP4DsUix5MhMPnXpiHDoKyoZ/bdCkwBCiQ=="], + "use-sync-external-store": ["use-sync-external-store@1.6.0", "", { "peerDependencies": { "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0" } }, "sha512-Pp6GSwGP/NrPIrxVFAIkOQeyw8lFenOHijQWkUTrDvrF4ALqylP2C/KCkeS9dpUM3KvYRQhna5vt7IL95+ZQ9w=="], + + "victory-vendor": ["victory-vendor@37.3.6", "", { "dependencies": { "@types/d3-array": "^3.0.3", "@types/d3-ease": "^3.0.0", "@types/d3-interpolate": "^3.0.1", "@types/d3-scale": "^4.0.2", "@types/d3-shape": "^3.1.0", "@types/d3-time": "^3.0.0", "@types/d3-timer": "^3.0.0", "d3-array": "^3.1.6", "d3-ease": "^3.0.1", "d3-interpolate": "^3.0.1", "d3-scale": "^4.0.2", "d3-shape": "^3.1.0", "d3-time": "^3.0.0", "d3-timer": "^3.0.1" } }, "sha512-SbPDPdDBYp+5MJHhBCAyI7wKM3d5ivekigc2Dk2s7pgbZ9wIgIBYGVw4zGHBml/qTFbexrofXW6Gu4noGxrOwQ=="], + "vite": ["vite@5.4.21", "", { "dependencies": { "esbuild": "^0.21.3", "postcss": "^8.4.43", "rollup": "^4.20.0" }, "optionalDependencies": { "fsevents": "~2.3.3" }, "peerDependencies": { "@types/node": "^18.0.0 || >=20.0.0", "less": "*", "lightningcss": "^1.21.0", "sass": "*", "sass-embedded": "*", "stylus": "*", "sugarss": "*", "terser": "^5.4.0" }, "optionalPeers": ["@types/node", "less", "lightningcss", "sass", "sass-embedded", "stylus", "sugarss", "terser"], "bin": { "vite": "bin/vite.js" } }, "sha512-o5a9xKjbtuhY6Bi5S3+HvbRERmouabWbyUcpXXUA1u+GNUKoROi9byOJ8M0nHbHYHkYICiMlqxkg1KkYmm25Sw=="], "vite-node": ["vite-node@2.1.8", "", { "dependencies": { "cac": "^6.7.14", "debug": "^4.3.7", "es-module-lexer": "^1.5.4", "pathe": "^1.1.2", "vite": "^5.0.0" }, "bin": { "vite-node": "vite-node.mjs" } }, "sha512-uPAwSr57kYjAUux+8E2j0q0Fxpn8M9VoyfGiRI8Kfktz9NcYMCenwY5RnZxnF1WTu3TGiYipirIzacLL3VVGFg=="], @@ -1202,6 +1279,8 @@ "@eslint-community/eslint-utils/eslint-visitor-keys": ["eslint-visitor-keys@3.4.3", "", {}, "sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag=="], + "@reduxjs/toolkit/immer": ["immer@11.1.4", "", {}, "sha512-XREFCPo6ksxVzP4E0ekD5aMdf8WMwmdNaz6vuvxgI40UaEiu6q3p8X52aU6GdyvLY3XXX/8R7JOTXStz/nBbRw=="], + "@tailwindcss/oxide-wasm32-wasi/@emnapi/core": ["@emnapi/core@1.9.1", "", { "dependencies": { "@emnapi/wasi-threads": "1.2.0", "tslib": "^2.4.0" }, "bundled": true }, "sha512-mukuNALVsoix/w1BJwFzwXBN/dHeejQtuVzcDsfOEsdpCumXb/E9j8w11h5S54tT1xhifGfbbSm/ICrObRb3KA=="], "@tailwindcss/oxide-wasm32-wasi/@emnapi/runtime": ["@emnapi/runtime@1.9.1", "", { "dependencies": { "tslib": "^2.4.0" }, "bundled": true }, "sha512-VYi5+ZVLhpgK4hQ0TAjiQiZ6ol0oe4mBx7mVv7IflsiEp0OWoVsp/+f9Vc1hOhE0TtkORVrI1GvzyreqpgWtkA=="], @@ -1254,7 +1333,7 @@ "pretty-format/ansi-styles": ["ansi-styles@5.2.0", "", {}, "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA=="], - "pretty-format/react-is": ["react-is@17.0.2", "", {}, "sha512-w2GsyukL62IJnlaff/nRegPQR94C/XXamvMWmSHRJ4y7Ts/4ocGRmTHvOs8PSE6pB3dWOrD/nueuU5sduBsQ4w=="], + "prop-types/react-is": ["react-is@16.13.1", "", {}, "sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ=="], "sharp/semver": ["semver@7.7.4", "", { "bin": { "semver": "bin/semver.js" } }, "sha512-vFKC2IEtQnVhpT78h1Yp8wzwrf8CM+MzKMHGJZfBtzhZNycRFnXsHk6E5TxIkkMsgNS7mdX3AGB7x2QM2di4lA=="], diff --git a/frontend/package.json b/frontend/package.json index 95641f4..732edbd 100644 --- a/frontend/package.json +++ b/frontend/package.json @@ -20,6 +20,7 @@ "react": "19.1.0", "react-dom": "19.1.0", "react-dropzone": "^15.0.0", + "recharts": "^3.8.1", "sonner": "^2.0.7" }, "devDependencies": { diff --git a/frontend/src/app/(app)/berichte/page.tsx b/frontend/src/app/(app)/berichte/page.tsx new file mode 100644 index 0000000..d395c3d --- /dev/null +++ b/frontend/src/app/(app)/berichte/page.tsx @@ -0,0 +1,262 @@ +"use client"; + +import { useState } from "react"; +import { useQuery } from "@tanstack/react-query"; +import { api } from "@/lib/api"; +import type { + CaseReport, + DeadlineReport, + WorkloadReport, + BillingReport, +} from "@/lib/types"; +import { Breadcrumb } from "@/components/layout/Breadcrumb"; +import { Skeleton } from "@/components/ui/Skeleton"; +import { + AlertTriangle, + RefreshCw, + Download, + Printer, + FolderOpen, + Clock, + Users, + Receipt, +} from "lucide-react"; +import { CasesTab } from "@/components/reports/CasesTab"; +import { DeadlinesTab } from "@/components/reports/DeadlinesTab"; +import { WorkloadTab } from "@/components/reports/WorkloadTab"; +import { BillingTab } from "@/components/reports/BillingTab"; + +type TabKey = "cases" | "deadlines" | "workload" | "billing"; + +const TABS: { key: TabKey; label: string; icon: typeof FolderOpen }[] = [ + { key: "cases", label: "Akten", icon: FolderOpen }, + { key: "deadlines", label: "Fristen", icon: Clock }, + { key: "workload", label: "Auslastung", icon: Users }, + { key: "billing", label: "Abrechnung", icon: Receipt }, +]; + +function getDefaultDateRange(): { from: string; to: string } { + const now = new Date(); + const from = new Date(now.getFullYear() - 1, now.getMonth(), 1); + return { + from: from.toISOString().split("T")[0], + to: now.toISOString().split("T")[0], + }; +} + +function ReportSkeleton() { + return ( +
+
+ {[1, 2, 3].map((i) => ( + + ))} +
+ + +
+ ); +} + +export default function BerichtePage() { + const [activeTab, setActiveTab] = useState("cases"); + const defaults = getDefaultDateRange(); + const [from, setFrom] = useState(defaults.from); + const [to, setTo] = useState(defaults.to); + + const queryParams = `?from=${from}&to=${to}`; + + const casesQuery = useQuery({ + queryKey: ["reports", "cases", from, to], + queryFn: () => api.get(`/reports/cases${queryParams}`), + enabled: activeTab === "cases", + }); + + const deadlinesQuery = useQuery({ + queryKey: ["reports", "deadlines", from, to], + queryFn: () => api.get(`/reports/deadlines${queryParams}`), + enabled: activeTab === "deadlines", + }); + + const workloadQuery = useQuery({ + queryKey: ["reports", "workload", from, to], + queryFn: () => api.get(`/reports/workload${queryParams}`), + enabled: activeTab === "workload", + }); + + const billingQuery = useQuery({ + queryKey: ["reports", "billing", from, to], + queryFn: () => api.get(`/reports/billing${queryParams}`), + enabled: activeTab === "billing", + }); + + const currentQuery = { + cases: casesQuery, + deadlines: deadlinesQuery, + workload: workloadQuery, + billing: billingQuery, + }[activeTab]; + + function exportCSV() { + if (!currentQuery.data) return; + let csv = ""; + const data = currentQuery.data; + + if (activeTab === "cases") { + const d = data as CaseReport; + csv = "Monat,Eroeffnet,Geschlossen,Aktiv\n"; + csv += d.monthly + .map((r) => `${r.period},${r.opened},${r.closed},${r.active}`) + .join("\n"); + } else if (activeTab === "deadlines") { + const d = data as DeadlineReport; + csv = "Monat,Gesamt,Eingehalten,Versaeumt,Ausstehend,Quote (%)\n"; + csv += d.monthly + .map( + (r) => + `${r.period},${r.total},${r.met},${r.missed},${r.pending},${r.compliance_rate.toFixed(1)}`, + ) + .join("\n"); + } else if (activeTab === "workload") { + const d = data as WorkloadReport; + csv = "Benutzer-ID,Aktive Akten,Fristen,Ueberfaellig,Erledigt\n"; + csv += d.users + .map( + (r) => + `${r.user_id},${r.active_cases},${r.deadlines},${r.overdue},${r.completed}`, + ) + .join("\n"); + } else if (activeTab === "billing") { + const d = data as BillingReport; + csv = "Monat,Aktiv,Geschlossen,Neu\n"; + csv += d.monthly + .map( + (r) => + `${r.period},${r.cases_active},${r.cases_closed},${r.cases_new}`, + ) + .join("\n"); + } + + const blob = new Blob([csv], { type: "text/csv;charset=utf-8;" }); + const url = URL.createObjectURL(blob); + const link = document.createElement("a"); + link.href = url; + link.download = `bericht-${activeTab}-${from}-${to}.csv`; + link.click(); + URL.revokeObjectURL(url); + } + + return ( +
+
+ +
+ +
+
+

Berichte

+

+ Statistiken und Auswertungen +

+
+ +
+
+ + setFrom(e.target.value)} + className="rounded-md border border-neutral-200 bg-white px-2.5 py-1.5 text-sm text-neutral-900 outline-none focus:border-neutral-400" + /> + + setTo(e.target.value)} + className="rounded-md border border-neutral-200 bg-white px-2.5 py-1.5 text-sm text-neutral-900 outline-none focus:border-neutral-400" + /> +
+ + + + +
+
+ + {/* Tabs */} +
+ +
+ + {/* Tab content */} + {currentQuery.isLoading && } + + {currentQuery.error && ( +
+
+ +
+

+ Bericht konnte nicht geladen werden +

+

+ Bitte versuchen Sie es erneut. +

+ +
+ )} + + {!currentQuery.isLoading && !currentQuery.error && currentQuery.data && ( + <> + {activeTab === "cases" && ( + + )} + {activeTab === "deadlines" && ( + + )} + {activeTab === "workload" && ( + + )} + {activeTab === "billing" && ( + + )} + + )} +
+ ); +} diff --git a/frontend/src/components/layout/Sidebar.tsx b/frontend/src/components/layout/Sidebar.tsx index 8896b17..ebd2664 100644 --- a/frontend/src/components/layout/Sidebar.tsx +++ b/frontend/src/components/layout/Sidebar.tsx @@ -8,6 +8,7 @@ import { Clock, Calendar, Brain, + BarChart3, Settings, Menu, X, @@ -27,6 +28,7 @@ const allNavigation: NavItem[] = [ { name: "Akten", href: "/cases", icon: FolderOpen }, { name: "Fristen", href: "/fristen", icon: Clock }, { name: "Termine", href: "/termine", icon: Calendar }, + { name: "Berichte", href: "/berichte", icon: BarChart3 }, { name: "AI Analyse", href: "/ai/extract", icon: Brain, permission: "ai_extraction" }, { name: "Einstellungen", href: "/einstellungen", icon: Settings, permission: "manage_settings" }, ]; diff --git a/frontend/src/components/reports/BillingTab.tsx b/frontend/src/components/reports/BillingTab.tsx new file mode 100644 index 0000000..4b8c631 --- /dev/null +++ b/frontend/src/components/reports/BillingTab.tsx @@ -0,0 +1,240 @@ +"use client"; + +import type { BillingReport } from "@/lib/types"; +import { + BarChart, + Bar, + XAxis, + YAxis, + CartesianGrid, + Tooltip, + ResponsiveContainer, + Legend, + LineChart, + Line, +} from "recharts"; +import { Receipt, TrendingUp, FolderOpen } from "lucide-react"; + +function formatMonth(period: string): string { + const [year, month] = period.split("-"); + const months = [ + "Jan", + "Feb", + "Mär", + "Apr", + "Mai", + "Jun", + "Jul", + "Aug", + "Sep", + "Okt", + "Nov", + "Dez", + ]; + return `${months[parseInt(month, 10) - 1]} ${year.slice(2)}`; +} + +export function BillingTab({ data }: { data: BillingReport }) { + const chartData = data.monthly.map((m) => ({ + ...m, + name: formatMonth(m.period), + })); + + const totalNew = data.monthly.reduce((sum, m) => sum + m.cases_new, 0); + const totalClosed = data.monthly.reduce((sum, m) => sum + m.cases_closed, 0); + const totalByType = data.by_type.reduce((sum, t) => sum + t.total, 0); + + return ( +
+ {/* Summary cards */} +
+
+
+ + Neue Mandate +
+

+ {totalNew} +

+

im Zeitraum

+
+
+
+ + Abgeschlossen +
+

+ {totalClosed} +

+

abrechenbar

+
+
+
+ + Verfahrensarten +
+

+ {data.by_type.length} +

+

+ {totalByType} Akten gesamt +

+
+
+ + {/* New cases trend */} +
+

+ Umsatzentwicklung (Mandate) +

+ {chartData.length === 0 ? ( +

+ Keine Daten im gewählten Zeitraum +

+ ) : ( + + + + + + + + + + + + )} +
+ + {/* By type breakdown */} +
+
+

+ Mandate nach Verfahrensart +

+ {data.by_type.length === 0 ? ( +

+ Keine Daten +

+ ) : ( + + + + + + + + + + + + )} +
+ + {/* Summary table */} +
+
+

+ Zusammenfassung +

+
+ {data.by_type.length === 0 ? ( +

+ Keine Daten +

+ ) : ( +
+ + + + + + + + + + + {data.by_type.map((t) => ( + + + + + + + ))} + +
VerfahrensartAktiv + Geschlossen + + Gesamt +
+ {t.case_type} + + {t.active} + + {t.closed} + + {t.total} +
+
+ )} +
+
+
+ ); +} diff --git a/frontend/src/components/reports/CasesTab.tsx b/frontend/src/components/reports/CasesTab.tsx new file mode 100644 index 0000000..4f5154f --- /dev/null +++ b/frontend/src/components/reports/CasesTab.tsx @@ -0,0 +1,223 @@ +"use client"; + +import type { CaseReport } from "@/lib/types"; +import { + BarChart, + Bar, + XAxis, + YAxis, + CartesianGrid, + Tooltip, + ResponsiveContainer, + Legend, + PieChart, + Pie, + Cell, +} from "recharts"; +import { FolderOpen, TrendingUp, TrendingDown } from "lucide-react"; + +const COLORS = [ + "#171717", + "#525252", + "#a3a3a3", + "#d4d4d4", + "#737373", + "#404040", + "#e5e5e5", + "#262626", +]; + +function formatMonth(period: string): string { + const [year, month] = period.split("-"); + const months = [ + "Jan", + "Feb", + "Mär", + "Apr", + "Mai", + "Jun", + "Jul", + "Aug", + "Sep", + "Okt", + "Nov", + "Dez", + ]; + return `${months[parseInt(month, 10) - 1]} ${year.slice(2)}`; +} + +export function CasesTab({ data }: { data: CaseReport }) { + const chartData = data.monthly.map((m) => ({ + ...m, + name: formatMonth(m.period), + })); + + return ( +
+ {/* Summary cards */} +
+
+
+ + Eröffnet +
+

+ {data.total.opened} +

+
+
+
+ + Geschlossen +
+

+ {data.total.closed} +

+
+
+
+ + Aktiv +
+

+ {data.total.active} +

+
+
+ + {/* Bar chart: opened/closed per month */} +
+

+ Akten pro Monat +

+ {chartData.length === 0 ? ( +

+ Keine Daten im gewählten Zeitraum +

+ ) : ( + + + + + + + + + + + + )} +
+ + {/* Pie charts row */} +
+ {/* By type */} +
+

+ Nach Verfahrensart +

+ {data.by_type.length === 0 ? ( +

+ Keine Daten +

+ ) : ( +
+ + + + {data.by_type.map((_, i) => ( + + ))} + + + + +
+ {data.by_type.map((item, i) => ( +
+
+ {item.case_type} + + {item.count} + +
+ ))} +
+
+ )} +
+ + {/* By court */} +
+

+ Nach Gericht +

+ {data.by_court.length === 0 ? ( +

+ Keine Daten +

+ ) : ( +
+ {data.by_court.map((item, i) => { + const maxCount = Math.max(...data.by_court.map((c) => c.count)); + const pct = maxCount > 0 ? (item.count / maxCount) * 100 : 0; + return ( +
+
+ {item.court} + + {item.count} + +
+
+
+
+
+ ); + })} +
+ )} +
+
+
+ ); +} diff --git a/frontend/src/components/reports/DeadlinesTab.tsx b/frontend/src/components/reports/DeadlinesTab.tsx new file mode 100644 index 0000000..ab5fd74 --- /dev/null +++ b/frontend/src/components/reports/DeadlinesTab.tsx @@ -0,0 +1,204 @@ +"use client"; + +import type { DeadlineReport } from "@/lib/types"; +import Link from "next/link"; +import { + LineChart, + Line, + XAxis, + YAxis, + CartesianGrid, + Tooltip, + ResponsiveContainer, + Legend, +} from "recharts"; +import { CheckCircle, XCircle, Clock, AlertTriangle } from "lucide-react"; + +function formatMonth(period: string): string { + const [year, month] = period.split("-"); + const months = [ + "Jan", + "Feb", + "Mär", + "Apr", + "Mai", + "Jun", + "Jul", + "Aug", + "Sep", + "Okt", + "Nov", + "Dez", + ]; + return `${months[parseInt(month, 10) - 1]} ${year.slice(2)}`; +} + +function formatDate(dateStr: string): string { + const d = new Date(dateStr); + return d.toLocaleDateString("de-DE", { + day: "2-digit", + month: "2-digit", + year: "numeric", + }); +} + +export function DeadlinesTab({ data }: { data: DeadlineReport }) { + const chartData = data.monthly.map((m) => ({ + ...m, + name: formatMonth(m.period), + compliance_rate: Math.round(m.compliance_rate * 10) / 10, + })); + + const complianceColor = + data.total.compliance_rate >= 90 + ? "text-emerald-600" + : data.total.compliance_rate >= 70 + ? "text-amber-600" + : "text-red-600"; + + return ( +
+ {/* Summary cards */} +
+
+
+ + Gesamt +
+

+ {data.total.total} +

+
+
+
+ + Eingehalten +
+

+ {data.total.met} +

+
+
+
+ + Versäumt +
+

+ {data.total.missed} +

+
+
+
+ + Einhaltungsquote +
+

+ {data.total.compliance_rate.toFixed(1)}% +

+
+
+ + {/* Compliance rate over time */} +
+

+ Fristeneinhaltung im Zeitverlauf +

+ {chartData.length === 0 ? ( +

+ Keine Daten im gewählten Zeitraum +

+ ) : ( + + + + + + [`${value}%`, "Quote"]} + /> + + + + + )} +
+ + {/* Missed deadlines table */} +
+
+

+ Versäumte Fristen +

+
+ {data.missed.length === 0 ? ( +
+ +

+ Keine versäumten Fristen im gewählten Zeitraum +

+
+ ) : ( +
+ + + + + + + + + + + {data.missed.map((d) => ( + + + + + + + ))} + +
FristAkteFällig am + Tage überfällig +
{d.title} + + {d.case_number} — {d.case_title} + + + {formatDate(d.due_date)} + + + + {d.days_overdue} + +
+
+ )} +
+
+ ); +} diff --git a/frontend/src/components/reports/WorkloadTab.tsx b/frontend/src/components/reports/WorkloadTab.tsx new file mode 100644 index 0000000..5124829 --- /dev/null +++ b/frontend/src/components/reports/WorkloadTab.tsx @@ -0,0 +1,187 @@ +"use client"; + +import type { WorkloadReport } from "@/lib/types"; +import { + BarChart, + Bar, + XAxis, + YAxis, + CartesianGrid, + Tooltip, + ResponsiveContainer, + Legend, +} from "recharts"; +import { Users, AlertTriangle, CheckCircle } from "lucide-react"; + +export function WorkloadTab({ data }: { data: WorkloadReport }) { + const chartData = data.users.map((u, i) => ({ + name: `Nutzer ${i + 1}`, + user_id: u.user_id, + active_cases: u.active_cases, + deadlines: u.deadlines, + overdue: u.overdue, + completed: u.completed, + })); + + const totalCases = data.users.reduce((sum, u) => sum + u.active_cases, 0); + const totalOverdue = data.users.reduce((sum, u) => sum + u.overdue, 0); + const totalCompleted = data.users.reduce((sum, u) => sum + u.completed, 0); + + return ( +
+ {/* Summary cards */} +
+
+
+ + Mitarbeiter +
+

+ {data.users.length} +

+

+ {totalCases} aktive Akten gesamt +

+
+
+
+ + Überfällige Fristen +
+

+ {totalOverdue} +

+
+
+
+ + Erledigte Fristen +
+

+ {totalCompleted} +

+
+
+ + {/* Stacked bar chart */} +
+

+ Auslastung pro Mitarbeiter +

+ {chartData.length === 0 ? ( +

+ Keine Daten im gewählten Zeitraum +

+ ) : ( + + + + + + + + + + + + + )} +
+ + {/* Table */} +
+
+

+ Übersicht pro Mitarbeiter +

+
+ {data.users.length === 0 ? ( +

+ Keine Mitarbeiter mit zugewiesenen Akten +

+ ) : ( +
+ + + + + + + + + + + + {data.users.map((u, i) => ( + + + + + + + + ))} + +
Mitarbeiter + Aktive Akten + Fristen + Überfällig + + Erledigt +
+ Nutzer {i + 1} + + {u.user_id.slice(0, 8)}... + + + {u.active_cases} + + {u.deadlines} + + {u.overdue > 0 ? ( + + {u.overdue} + + ) : ( + 0 + )} + + {u.completed} +
+
+ )} +
+
+ ); +} diff --git a/frontend/src/lib/types.ts b/frontend/src/lib/types.ts index a842be4..3000b5a 100644 --- a/frontend/src/lib/types.ts +++ b/frontend/src/lib/types.ts @@ -329,3 +329,149 @@ export interface ExtractionResponse { deadlines: ExtractedDeadline[]; count: number; } + +// Notification types + +export interface Notification { + id: string; + tenant_id: string; + user_id: string; + type: string; + entity_type?: string; + entity_id?: string; + title: string; + body?: string; + sent_at?: string; + read_at?: string; + created_at: string; +} + +export interface NotificationPreferences { + user_id: string; + tenant_id: string; + deadline_reminder_days: number[]; + email_enabled: boolean; + daily_digest: boolean; + created_at: string; + updated_at: string; +} + +export interface NotificationListResponse { + data: Notification[]; + total: number; +} + +// Audit log types + +export interface AuditLogEntry { + id: number; + tenant_id: string; + user_id?: string; + action: string; + entity_type: string; + entity_id?: string; + old_values?: Record; + new_values?: Record; + ip_address?: string; + user_agent?: string; + created_at: string; +} + +export interface AuditLogResponse { + entries: AuditLogEntry[]; + total: number; + page: number; + limit: number; +} + +// Reporting types + +export interface CaseStats { + period: string; + opened: number; + closed: number; + active: number; +} + +export interface CasesByType { + case_type: string; + count: number; +} + +export interface CasesByCourt { + court: string; + count: number; +} + +export interface CaseReport { + monthly: CaseStats[]; + by_type: CasesByType[]; + by_court: CasesByCourt[]; + total: { + opened: number; + closed: number; + active: number; + }; +} + +export interface DeadlineCompliance { + period: string; + total: number; + met: number; + missed: number; + pending: number; + compliance_rate: number; +} + +export interface MissedDeadline { + id: string; + title: string; + due_date: string; + case_id: string; + case_number: string; + case_title: string; + days_overdue: number; +} + +export interface DeadlineReport { + monthly: DeadlineCompliance[]; + missed: MissedDeadline[]; + total: { + total: number; + met: number; + missed: number; + pending: number; + compliance_rate: number; + }; +} + +export interface UserWorkload { + user_id: string; + active_cases: number; + deadlines: number; + overdue: number; + completed: number; +} + +export interface WorkloadReport { + users: UserWorkload[]; +} + +export interface BillingByMonth { + period: string; + cases_active: number; + cases_closed: number; + cases_new: number; +} + +export interface BillingByType { + case_type: string; + active: number; + closed: number; + total: number; +} + +export interface BillingReport { + monthly: BillingByMonth[]; + by_type: BillingByType[]; +}