2026-03-06 11:33:32 -06:00
|
|
|
|
# CPAS Violation Tracker
|
|
|
|
|
|
|
2026-03-06 23:18:28 -06:00
|
|
|
|
Single-container Dockerized web app for CPAS violation documentation and workforce standing management.
|
|
|
|
|
|
Built with **React + Vite** (frontend), **Node.js + Express** (backend), **SQLite** (database), and **Puppeteer** (PDF generation).
|
|
|
|
|
|
|
|
|
|
|
|
---
|
2026-03-06 11:33:32 -06:00
|
|
|
|
|
|
|
|
|
|
## The only requirement on your machine: Docker Desktop
|
|
|
|
|
|
|
2026-03-06 23:18:28 -06:00
|
|
|
|
Everything else — Node.js, npm, React build, Chromium for PDF — happens inside Docker.
|
2026-03-06 11:33:32 -06:00
|
|
|
|
|
|
|
|
|
|
---
|
|
|
|
|
|
|
|
|
|
|
|
## Quickstart (Local)
|
|
|
|
|
|
|
|
|
|
|
|
```bash
|
|
|
|
|
|
# 1. Build the image (installs all deps + compiles React inside Docker)
|
|
|
|
|
|
docker build -t cpas-tracker .
|
|
|
|
|
|
|
|
|
|
|
|
# 2. Run it
|
|
|
|
|
|
docker run -d --name cpas-tracker \
|
|
|
|
|
|
-p 3001:3001 \
|
|
|
|
|
|
-v cpas-data:/data \
|
|
|
|
|
|
cpas-tracker
|
|
|
|
|
|
|
|
|
|
|
|
# 3. Open
|
|
|
|
|
|
# http://localhost:3001
|
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
|
|
## Export for Unraid
|
|
|
|
|
|
|
|
|
|
|
|
```bash
|
|
|
|
|
|
docker save cpas-tracker | gzip > cpas-tracker.tar.gz
|
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
|
|
Then follow README_UNRAID_INSTALL.md.
|
|
|
|
|
|
|
|
|
|
|
|
## Update After Code Changes
|
|
|
|
|
|
|
|
|
|
|
|
```bash
|
|
|
|
|
|
docker build -t cpas-tracker .
|
|
|
|
|
|
docker stop cpas-tracker && docker rm cpas-tracker
|
|
|
|
|
|
docker run -d --name cpas-tracker -p 3001:3001 -v cpas-data:/data cpas-tracker
|
|
|
|
|
|
```
|
|
|
|
|
|
|
2026-03-06 23:18:28 -06:00
|
|
|
|
---
|
|
|
|
|
|
|
|
|
|
|
|
## Features
|
|
|
|
|
|
|
|
|
|
|
|
### Company Dashboard
|
|
|
|
|
|
- Live table of all employees sorted by active CPAS points (highest risk first)
|
|
|
|
|
|
- Summary stat cards: total employees, elite standing (0 pts), with active points, at-risk count, highest active score
|
|
|
|
|
|
- **At-risk badge**: flags employees within 2 points of the next tier escalation
|
|
|
|
|
|
- Search/filter by name, department, or supervisor
|
|
|
|
|
|
- Click any employee name to open their full profile modal
|
|
|
|
|
|
|
|
|
|
|
|
### Violation Form
|
|
|
|
|
|
- Select existing employee or enter new employee by name
|
|
|
|
|
|
- **Employee intelligence**: shows current CPAS standing badge and 90-day violation count before submitting
|
|
|
|
|
|
- Violation type dropdown grouped by category; shows prior 90-day counts inline
|
|
|
|
|
|
- **Recidivist auto-escalation**: if an employee has prior violations of the same type, points slider auto-sets to maximum per policy
|
|
|
|
|
|
- Repeat offense badge with prior count displayed
|
|
|
|
|
|
- Context-sensitive fields (time, minutes late, amount, location, description) shown only when relevant to violation type
|
|
|
|
|
|
- **Tier crossing warning** (TierWarning component): previews what tier the new points would push the employee into before submission
|
|
|
|
|
|
- Point slider for discretionary adjustments within the violation's min/max range
|
|
|
|
|
|
- One-click PDF download immediately after submission
|
|
|
|
|
|
|
|
|
|
|
|
### Employee Profile Modal
|
|
|
|
|
|
- Full violation history with resolution status
|
|
|
|
|
|
- Negate / restore individual violations (soft delete with resolution type + notes)
|
|
|
|
|
|
- Hard delete option for data entry errors
|
|
|
|
|
|
- PDF download for any historical violation record
|
|
|
|
|
|
|
|
|
|
|
|
### CPAS Tier System
|
|
|
|
|
|
|
|
|
|
|
|
| Points | Tier | Label |
|
|
|
|
|
|
|--------|------|-------|
|
|
|
|
|
|
| 0–4 | 0–1 | Elite Standing |
|
|
|
|
|
|
| 5–9 | 1 | Realignment |
|
|
|
|
|
|
| 10–14 | 2 | Administrative Lockdown |
|
|
|
|
|
|
| 15–19 | 3 | Verification |
|
|
|
|
|
|
| 20–24 | 4 | Risk Mitigation |
|
|
|
|
|
|
| 25–29 | 5 | Final Decision |
|
|
|
|
|
|
| 30+ | 6 | Separation |
|
|
|
|
|
|
|
|
|
|
|
|
Scores are computed over a **rolling 90-day window** (negated violations excluded).
|
|
|
|
|
|
|
|
|
|
|
|
### PDF Generation
|
|
|
|
|
|
- Puppeteer + system Chromium (bundled in Docker image)
|
|
|
|
|
|
- Generated on-demand per violation via `GET /api/violations/:id/pdf`
|
|
|
|
|
|
- Filename: `CPAS_<EmployeeName>_<IncidentDate>.pdf`
|
|
|
|
|
|
- PDF captures prior active points **at the time of the incident** (snapshot stored on insert)
|
|
|
|
|
|
|
|
|
|
|
|
---
|
|
|
|
|
|
|
|
|
|
|
|
## API Reference
|
|
|
|
|
|
|
|
|
|
|
|
| Method | Endpoint | Description |
|
|
|
|
|
|
|--------|----------|-------------|
|
|
|
|
|
|
| GET | `/api/health` | Health check |
|
|
|
|
|
|
| GET | `/api/employees` | List all employees |
|
|
|
|
|
|
| POST | `/api/employees` | Create or upsert employee |
|
|
|
|
|
|
| GET | `/api/employees/:id/score` | Get active CPAS score for employee |
|
|
|
|
|
|
| GET | `/api/dashboard` | All employees with active points + violation counts |
|
|
|
|
|
|
| POST | `/api/violations` | Log a new violation |
|
|
|
|
|
|
| GET | `/api/violations/employee/:id` | Get violation history for employee (with resolutions) |
|
|
|
|
|
|
| PATCH | `/api/violations/:id/negate` | Negate a violation (soft delete + resolution record) |
|
|
|
|
|
|
| PATCH | `/api/violations/:id/restore` | Restore a negated violation |
|
|
|
|
|
|
| DELETE | `/api/violations/:id` | Hard delete a violation |
|
|
|
|
|
|
| GET | `/api/violations/:id/pdf` | Download violation PDF |
|
|
|
|
|
|
|
|
|
|
|
|
---
|
|
|
|
|
|
|
2026-03-06 11:33:32 -06:00
|
|
|
|
## Project Structure
|
|
|
|
|
|
|
|
|
|
|
|
```
|
2026-03-06 23:18:28 -06:00
|
|
|
|
cpas/
|
|
|
|
|
|
├── Dockerfile # Multi-stage: builds React + runs Express w/ Chromium
|
2026-03-06 11:33:32 -06:00
|
|
|
|
├── .dockerignore
|
2026-03-06 23:18:28 -06:00
|
|
|
|
├── package.json # Backend (Express) deps
|
|
|
|
|
|
├── server.js # API + static file server
|
2026-03-06 11:33:32 -06:00
|
|
|
|
├── db/
|
2026-03-06 23:18:28 -06:00
|
|
|
|
│ ├── schema.sql # Tables + 90-day active score view
|
|
|
|
|
|
│ └── database.js # SQLite connection (better-sqlite3)
|
|
|
|
|
|
├── pdf/
|
|
|
|
|
|
│ └── generator.js # Puppeteer PDF generation
|
|
|
|
|
|
└── client/ # React frontend (Vite)
|
2026-03-06 11:33:32 -06:00
|
|
|
|
├── package.json
|
|
|
|
|
|
├── vite.config.js
|
|
|
|
|
|
├── index.html
|
|
|
|
|
|
└── src/
|
|
|
|
|
|
├── main.jsx
|
|
|
|
|
|
├── App.jsx
|
|
|
|
|
|
├── data/
|
2026-03-06 23:18:28 -06:00
|
|
|
|
│ └── violations.js # All CPAS violation definitions + groups
|
|
|
|
|
|
├── hooks/
|
|
|
|
|
|
│ └── useEmployeeIntelligence.js # Score + history hook
|
2026-03-06 11:33:32 -06:00
|
|
|
|
└── components/
|
2026-03-06 23:18:28 -06:00
|
|
|
|
├── CpasBadge.jsx # Tier badge + color logic
|
|
|
|
|
|
├── TierWarning.jsx # Pre-submit tier crossing alert
|
|
|
|
|
|
├── Dashboard.jsx # Company-wide leaderboard
|
|
|
|
|
|
├── ViolationForm.jsx # Violation entry form
|
|
|
|
|
|
├── EmployeeModal.jsx # Employee profile + history modal
|
|
|
|
|
|
├── NegateModal.jsx # Negate/resolve violation dialog
|
|
|
|
|
|
└── ViolationHistory.jsx # Violation list component
|
2026-03-06 11:33:32 -06:00
|
|
|
|
```
|
|
|
|
|
|
|
2026-03-06 23:18:28 -06:00
|
|
|
|
---
|
|
|
|
|
|
|
|
|
|
|
|
## Database Schema
|
|
|
|
|
|
|
|
|
|
|
|
Three tables + one view:
|
|
|
|
|
|
|
|
|
|
|
|
- **`employees`** — id, name, department, supervisor
|
|
|
|
|
|
- **`violations`** — full incident record including `prior_active_points` snapshot at time of logging
|
|
|
|
|
|
- **`violation_resolutions`** — resolution type, details, resolved_by (linked to violations)
|
|
|
|
|
|
- **`active_cpas_scores`** (view) — sum of points for non-negated violations in rolling 90 days, grouped by employee
|
|
|
|
|
|
|
|
|
|
|
|
---
|
|
|
|
|
|
|
2026-03-06 11:33:32 -06:00
|
|
|
|
## Phases
|
2026-03-06 23:18:28 -06:00
|
|
|
|
|
2026-03-06 11:33:32 -06:00
|
|
|
|
- [x] Phase 1 — Container scaffold, SQLite schema, base React form
|
2026-03-06 23:18:28 -06:00
|
|
|
|
- [x] Phase 2 — Employee history, prior violation highlighting, recidivist point auto-suggest
|
|
|
|
|
|
- [x] Phase 3 — Puppeteer PDF generation
|
|
|
|
|
|
- [x] Phase 4 — Dashboard, CPAS scores, tier warnings, at-risk badges
|
|
|
|
|
|
- [ ] Phase 5 — Recidivist point auto-suggest refinements / additional reporting
|