This file provides guidance to Claude Code (claude.ai/code) when working with code in this repository.
Checkmate is an open-source uptime and infrastructure monitoring application. It monitors server hardware, uptime, response times, and incidents with real-time alerts. The companion agent Capture provides infrastructure metrics (CPU, RAM, disk, temperature).
cd client
npm install
npm run dev # Start dev server at http://localhost:5173
npm run build # TypeScript check + production build
npm run lint # ESLint (strict, max-warnings 0)
npm run format # Prettier formatting
npm run format-check # Check formattingcd server
npm install
npm run dev # Start with hot-reload (nodemon + tsx) at http://localhost:52345
npm run build # TypeScript compile + path alias resolution
npm run test # Run Mocha tests with c8 coverage
npm run lint # ESLint v9
npm run lint-fix # Auto-fix lint issues
npm run format # Prettier formattingcd docker/dev
./build_images.sh
docker run -d -p 27017:27017 -v uptime_mongo_data:/data/db --name uptime_database_mongo mongo:6.0CLIENT_HOST="http://localhost:5173"
JWT_SECRET="my_secret_key_change_this"
DB_CONNECTION_STRING="mongodb://localhost:27017/uptime_db"
TOKEN_TTL="99d"
ORIGIN="localhost"
LOG_LEVEL="debug"VITE_APP_API_BASE_URL="http://localhost:52345/api/v1"
VITE_APP_LOG_LEVEL="debug"/client- React 18 + TypeScript + Vite + MUI frontend/server- Node.js 20+ + Express + TypeScript backend/docker- Multi-environment Docker configs (dev, staging, prod, arm, mono)
server/src/
├── controllers/ # Route handlers (authController, monitorController, etc.)
├── service/ # Business logic
│ ├── business/ # Core monitoring logic
│ ├── infrastructure/ # Server/system utilities
│ └── system/ # App-level settings
├── db/
│ ├── models/ # Mongoose schemas (Monitor, Check, Incident, User, etc.)
│ ├── migration/ # Database migrations (run on startup)
│ └── modules/ # Database-specific modules
├── middleware/v1/ # verifyJWT, rateLimiter, sanitization, responseHandler
├── routes/v1/ # API route definitions
├── validation/ # Joi input validation schemas
└── repositories/ # Data access layer
client/src/
├── Components/ # Reusable UI components
├── Pages/ # Page components (Auth, Uptime, Infrastructure, Incidents, etc.)
├── Features/ # Redux slices (Auth, UI)
├── Hooks/ # Custom React hooks
├── Utils/ # Utilities (NetworkService.js is main API client)
├── Validation/ # Input validation
└── locales/ # i18n translations
- Base URL:
/api/v1 - Documentation:
http://localhost:52345/api-docs(Swagger UI) - OpenAPI spec:
/server/openapi.json
- State Management: Redux Toolkit + Redux-Persist
- Data Fetching: SWR + Axios
- Database: MongoDB with Mongoose ODM
- Queue/Cache: Redis + BullMQ + Pulse (cron scheduling)
- i18n: i18next + react-i18next (translations via PoEditor)
The backend enforces a strict three-layer separation between HTTP handling, business logic, and data access:
Request → Controller → Service → Repository → MongoDB (Mongoose)
- Controllers (
/controllers) handle HTTP concerns only: parsing request params, calling the appropriate service, and returning a response via theresponseHandlermiddleware. They contain no business logic. - Services (
/service/business) contain all business logic: deciding whether an incident should be created, whether a notification should fire, what state a monitor is in, etc. - Repositories (
/repositories) are the sole layer that talks to MongoDB through Mongoose. They expose clean, reusable query methods (e.g.findByMonitorId,createCheck) so that services never construct raw DB queries directly.
This separation makes each layer independently testable and keeps Mongoose-specific code out of business logic. When adding a new feature, the pattern to follow is: add a repository method for any new DB query, call it from a service, and expose it via a controller route.
Background monitoring runs on a scheduled queue, not on the HTTP request cycle. The high-level flow for uptime monitoring is:
Pulse (cron) → BullMQ Job → StatusService
├── performs HTTP/port/ping check
├── saves Check via CheckRepository
├── evaluates monitor state change
│ └── calls IncidentService (create / resolve incident)
└── calls NotificationService (email, Slack, Discord, webhook)
- Pulse (cron scheduler) enqueues a job into a BullMQ queue for each active monitor at its configured interval.
- A BullMQ worker picks up the job and calls
StatusService, which performs the actual check (HTTP request, TCP port probe, ping, etc.). - The result is persisted as a
Checkdocument via the repository layer. StatusServicecompares the new result against the monitor's previous state. If the monitor transitions from up → down (or down → up), it delegates toIncidentServiceto open or resolve anIncidentdocument.- On a state change,
NotificationServicereads the monitor's configuredNotificationdocuments and dispatches alerts to all enabled channels (email, Discord, Slack, webhooks).
Redis serves two roles: job queue storage for BullMQ and ephemeral caching. BullMQ manages concurrency, retries, and backpressure for monitoring jobs, ensuring checks are processed reliably even under load.
- Each monitor type (HTTP, port, ping, infrastructure) maps to its own queue worker so failures in one type don't block others.
- Job scheduling interval is driven by the
intervalfield on theMonitormodel. - Failed jobs are retried with configurable backoff before being moved to a dead-letter state.
- Redis is also used to cache frequently read data (e.g. aggregated stats) to reduce MongoDB query pressure.
When working on anything related to check scheduling, incident lifecycle, or notifications, trace the flow starting from the relevant BullMQ worker rather than from the controller layer.
All user-facing strings must use the translation function:
t('your.key') // Never hardcode UI strings- Always branch from
develop(not master) - Use descriptive names:
feat/add-alerts,fix/login-error - PRs target
developbranch
- Client: Prettier with
printWidth: 90, tabs, double quotes - Server: Prettier with
printWidth: 150, tabs, double quotes - Both use ESLint with strict settings
Server tests use Mocha + Chai + Sinon:
npm test # Run all tests with coverage
npm test -- --grep "pattern" # Run specific testsTest files: server/tests/**/*.test.js
Key Mongoose models in /server/src/db/models/:
- Monitor - Monitoring configuration (website, infrastructure, port, etc.)
- Check - Individual monitoring check results
- Incident - Downtime incidents
- User - User accounts
- Team - Team/workspace management
- StatusPage - Public status pages
- Notification - Alert configuration (email, Discord, Slack, webhooks)
- MaintenanceWindow - Scheduled maintenance periods
- AppSettings - Global application settings