diff --git a/backend/internal/auth/context.go b/backend/internal/auth/context.go index 333190c..9553104 100644 --- a/backend/internal/auth/context.go +++ b/backend/internal/auth/context.go @@ -14,19 +14,6 @@ const ( userRoleKey contextKey = "user_role" ipKey contextKey = "ip_address" userAgentKey contextKey = "user_agent" -<<<<<<< HEAD -||||||| 8e65463 -||||||| 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 -======= - userRoleKey contextKey = "user_role" ->>>>>>> mai/ritchie/p1-document-templates ) func ContextWithUserID(ctx context.Context, userID uuid.UUID) context.Context { @@ -46,7 +33,6 @@ 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) @@ -56,10 +42,6 @@ func UserRoleFromContext(ctx context.Context) string { role, _ := ctx.Value(userRoleKey).(string) return role } -||||||| 8e65463 -<<<<<<< HEAD -======= ->>>>>>> mai/ritchie/p1-document-templates func ContextWithRequestInfo(ctx context.Context, ip, userAgent string) context.Context { ctx = context.WithValue(ctx, ipKey, ip) @@ -80,28 +62,3 @@ func UserAgentFromContext(ctx context.Context) *string { } return nil } -<<<<<<< HEAD -||||||| 8e65463 -||||||| 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 -======= - -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/ritchie/p1-document-templates diff --git a/backend/internal/auth/middleware.go b/backend/internal/auth/middleware.go index fa25c98..804f770 100644 --- a/backend/internal/auth/middleware.go +++ b/backend/internal/auth/middleware.go @@ -35,6 +35,8 @@ func (m *Middleware) RequireAuth(next http.Handler) http.Handler { } ctx := ContextWithUserID(r.Context(), userID) + // Tenant resolution is handled by TenantResolver middleware for scoped routes. + // Tenant management routes handle their own access control. // Capture IP and user-agent for audit logging ip := r.Header.Get("X-Forwarded-For") @@ -43,9 +45,6 @@ func (m *Middleware) RequireAuth(next http.Handler) http.Handler { } ctx = ContextWithRequestInfo(ctx, ip, r.UserAgent()) - // Tenant resolution is handled by TenantResolver middleware for scoped routes. - // Tenant management routes handle their own access control. - next.ServeHTTP(w, r.WithContext(ctx)) }) } diff --git a/backend/internal/handlers/deadlines.go b/backend/internal/handlers/deadlines.go index 63c48d4..1ff7629 100644 --- a/backend/internal/handlers/deadlines.go +++ b/backend/internal/handlers/deadlines.go @@ -198,27 +198,8 @@ func (h *DeadlineHandlers) Delete(w http.ResponseWriter, r *http.Request) { return } -<<<<<<< HEAD - err = h.deadlines.Delete(r.Context(), tenantID, deadlineID) - if err != nil { - writeError(w, http.StatusNotFound, err.Error()) -||||||| 8e65463 -<<<<<<< HEAD - if err := h.deadlines.Delete(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 -======= if err := h.deadlines.Delete(r.Context(), tenantID, deadlineID); err != nil { writeError(w, http.StatusNotFound, "deadline not found") ->>>>>>> mai/ritchie/p1-document-templates 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 250ef8d..af868d2 100644 --- a/backend/internal/router/router.go +++ b/backend/internal/router/router.go @@ -31,15 +31,6 @@ func New(db *sqlx.DB, authMW *auth.Middleware, cfg *config.Config, calDAVSvc *se storageCli := services.NewStorageClient(cfg.SupabaseURL, cfg.SupabaseServiceKey) documentSvc := services.NewDocumentService(db, storageCli, auditSvc) assignmentSvc := services.NewCaseAssignmentService(db) -<<<<<<< HEAD - timeEntrySvc := services.NewTimeEntryService(db, auditSvc) - billingRateSvc := services.NewBillingRateService(db, auditSvc) - invoiceSvc := services.NewInvoiceService(db, auditSvc) -||||||| 8e65463 ->>>>>>> mai/pike/p0-role-based -======= - templateSvc := services.NewTemplateService(db, auditSvc) ->>>>>>> mai/ritchie/p1-document-templates // AI service (optional — only if API key is configured) var aiH *handlers.AIHandler @@ -53,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 @@ -70,18 +62,11 @@ 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) assignmentH := handlers.NewCaseAssignmentHandler(assignmentSvc) -<<<<<<< HEAD - timeEntryH := handlers.NewTimeEntryHandler(timeEntrySvc) - billingRateH := handlers.NewBillingRateHandler(billingRateSvc) - invoiceH := handlers.NewInvoiceHandler(invoiceSvc) -||||||| 8e65463 -======= - templateH := handlers.NewTemplateHandler(templateSvc, caseSvc, partySvc, deadlineSvc, tenantSvc) ->>>>>>> mai/ritchie/p1-document-templates // Public routes mux.HandleFunc("GET /health", handleHealth(db)) @@ -123,7 +108,7 @@ func New(db *sqlx.DB, authMW *auth.Middleware, cfg *config.Config, calDAVSvc *se scoped.HandleFunc("GET /api/cases", caseH.List) scoped.HandleFunc("POST /api/cases", perm(auth.PermCreateCase, caseH.Create)) scoped.HandleFunc("GET /api/cases/{id}", caseH.Get) - scoped.HandleFunc("PUT /api/cases/{id}", caseH.Update) + scoped.HandleFunc("PUT /api/cases/{id}", caseH.Update) // case-level access checked in handler scoped.HandleFunc("DELETE /api/cases/{id}", perm(auth.PermCreateCase, caseH.Delete)) // Parties — same access as case editing @@ -149,7 +134,7 @@ func New(db *sqlx.DB, authMW *auth.Middleware, cfg *config.Config, calDAVSvc *se // Deadline calculator — all can use scoped.HandleFunc("POST /api/deadlines/calculate", calcH.Calculate) - // Appointments — all can manage + // Appointments — all can manage (PermManageAppointments granted to all) scoped.HandleFunc("GET /api/appointments/{id}", apptH.Get) scoped.HandleFunc("GET /api/appointments", apptH.List) scoped.HandleFunc("POST /api/appointments", perm(auth.PermManageAppointments, apptH.Create)) @@ -173,49 +158,21 @@ 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 - // Audit log — view requires PermViewAuditLog - scoped.HandleFunc("GET /api/audit-log", perm(auth.PermViewAuditLog, auditH.List)) -||||||| 8e65463 -<<<<<<< HEAD - // Audit log - scoped.HandleFunc("GET /api/audit-log", auditH.List) -======= - // Audit log - scoped.HandleFunc("GET /api/audit-log", auditH.List) ->>>>>>> mai/ritchie/p1-document-templates + // 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) -<<<<<<< HEAD // Documents — all can upload, delete checked in handler (own vs all) -||||||| 8e65463 - // Documents -||||||| 82878df - // Documents -======= - // Documents — all can upload, delete checked in handler (own vs all) ->>>>>>> mai/pike/p0-role-based -======= - // Documents — all can upload, delete checked in handler ->>>>>>> mai/ritchie/p1-document-templates 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) scoped.HandleFunc("GET /api/documents/{docId}/meta", docH.GetMeta) -<<<<<<< HEAD - scoped.HandleFunc("DELETE /api/documents/{docId}", docH.Delete) -||||||| 8e65463 scoped.HandleFunc("DELETE /api/documents/{docId}", docH.Delete) // permission check inside handler -======= - scoped.HandleFunc("DELETE /api/documents/{docId}", docH.Delete) - - // Document templates — all can view, create/edit needs PermCreateCase - scoped.HandleFunc("GET /api/templates", templateH.List) - scoped.HandleFunc("GET /api/templates/{id}", templateH.Get) - scoped.HandleFunc("POST /api/templates", perm(auth.PermCreateCase, templateH.Create)) - scoped.HandleFunc("PUT /api/templates/{id}", perm(auth.PermCreateCase, templateH.Update)) - scoped.HandleFunc("DELETE /api/templates/{id}", perm(auth.PermCreateCase, templateH.Delete)) - scoped.HandleFunc("POST /api/templates/{id}/render", templateH.Render) ->>>>>>> mai/ritchie/p1-document-templates // AI endpoints (rate limited: 5 req/min burst 10 per IP) if aiH != nil { @@ -234,43 +191,13 @@ func New(db *sqlx.DB, authMW *auth.Middleware, cfg *config.Config, calDAVSvc *se scoped.HandleFunc("PUT /api/notification-preferences", notifH.UpdatePreferences) } -<<<<<<< HEAD // CalDAV sync endpoints — settings permission required -||||||| 8e65463 - // CalDAV sync endpoints -||||||| 82878df - // CalDAV sync endpoints -======= - // CalDAV sync endpoints — settings permission required ->>>>>>> mai/pike/p0-role-based -======= - // CalDAV sync endpoints ->>>>>>> mai/ritchie/p1-document-templates if calDAVSvc != nil { calDAVH := handlers.NewCalDAVHandler(calDAVSvc) scoped.HandleFunc("POST /api/caldav/sync", perm(auth.PermManageSettings, calDAVH.TriggerSync)) scoped.HandleFunc("GET /api/caldav/status", calDAVH.GetStatus) } - // Time entries — billing permission for create/update/delete - scoped.HandleFunc("GET /api/cases/{id}/time-entries", timeEntryH.ListForCase) - scoped.HandleFunc("POST /api/cases/{id}/time-entries", perm(auth.PermManageBilling, timeEntryH.Create)) - scoped.HandleFunc("GET /api/time-entries", timeEntryH.List) - scoped.HandleFunc("GET /api/time-entries/summary", perm(auth.PermManageBilling, timeEntryH.Summary)) - scoped.HandleFunc("PUT /api/time-entries/{id}", perm(auth.PermManageBilling, timeEntryH.Update)) - scoped.HandleFunc("DELETE /api/time-entries/{id}", perm(auth.PermManageBilling, timeEntryH.Delete)) - - // Billing rates — billing permission required - scoped.HandleFunc("GET /api/billing-rates", perm(auth.PermManageBilling, billingRateH.List)) - scoped.HandleFunc("PUT /api/billing-rates", perm(auth.PermManageBilling, billingRateH.Upsert)) - - // Invoices — billing permission required - scoped.HandleFunc("GET /api/invoices", perm(auth.PermManageBilling, invoiceH.List)) - scoped.HandleFunc("POST /api/invoices", perm(auth.PermManageBilling, invoiceH.Create)) - scoped.HandleFunc("GET /api/invoices/{id}", perm(auth.PermManageBilling, invoiceH.Get)) - scoped.HandleFunc("PUT /api/invoices/{id}", perm(auth.PermManageBilling, invoiceH.Update)) - scoped.HandleFunc("PATCH /api/invoices/{id}/status", perm(auth.PermManageBilling, invoiceH.UpdateStatus)) - // Wire: auth -> tenant routes go directly, scoped routes get tenant resolver api.Handle("/api/", tenantResolver.Resolve(scoped)) 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 ( +
+ Statistiken und Auswertungen +
++ Bitte versuchen Sie es erneut. +
+ ++ {totalNew} +
+im Zeitraum
++ {totalClosed} +
+abrechenbar
++ {data.by_type.length} +
++ {totalByType} Akten gesamt +
++ Keine Daten im gewählten Zeitraum +
+ ) : ( ++ Keine Daten +
+ ) : ( ++ Keine Daten +
+ ) : ( +| Verfahrensart | +Aktiv | ++ Geschlossen + | ++ Gesamt + | +
|---|---|---|---|
| + {t.case_type} + | ++ {t.active} + | ++ {t.closed} + | ++ {t.total} + | +
+ {data.total.opened} +
++ {data.total.closed} +
++ {data.total.active} +
++ Keine Daten im gewählten Zeitraum +
+ ) : ( ++ Keine Daten +
+ ) : ( ++ Keine Daten +
+ ) : ( ++ {data.total.total} +
++ {data.total.met} +
++ {data.total.missed} +
++ {data.total.compliance_rate.toFixed(1)}% +
++ Keine Daten im gewählten Zeitraum +
+ ) : ( ++ Keine versäumten Fristen im gewählten Zeitraum +
+| Frist | +Akte | +Fällig am | ++ Tage überfällig + | +
|---|---|---|---|
| {d.title} | ++ + {d.case_number} — {d.case_title} + + | ++ {formatDate(d.due_date)} + | +
+
+ |
+
+ {data.users.length} +
++ {totalCases} aktive Akten gesamt +
++ {totalOverdue} +
++ {totalCompleted} +
++ Keine Daten im gewählten Zeitraum +
+ ) : ( ++ Keine Mitarbeiter mit zugewiesenen Akten +
+ ) : ( +| 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} + | +