Local Development
Compass has multiple workable development modes. Pick the lightest mode that supports the feature you are changing.
Frontend-Only Mode
Command:
bun run dev:web
Use this for:
- most layout and interaction work
- local task behavior
- many event UI changes
- router and view work
You do not need the backend for basic frontend rendering.
Backend Mode
Command:
bun run dev:backend
Use this for:
- authenticated API work
- Google OAuth/session behavior
- Mongo-backed event behavior
- sync and SSE work
This requires valid env config.
Bootstrap once from repo root:
cp packages/backend/.env.local.example packages/backend/.env.local
Runtime note:
bun run dev:backend,bun run dev:web, andbun run cli ...load variables frompackages/backend/.env.localthrough Bun's--env-file.
Backend Environment Contract
Source:
packages/backend/src/common/constants/env.constants.ts
The backend validates env at startup with Zod.
Important variables:
NODE_ENVTZBASEURLPORTMONGO_URISUPERTOKENS_URISUPERTOKENS_KEYTOKEN_COMPASS_SYNCFRONTEND_URLCORS(parsed intoENV.ORIGINS_ALLOWED)
Optional but behavior-changing:
GOOGLE_CLIENT_IDGOOGLE_CLIENT_SECRETGCAL_WEBHOOK_BASEURLTOKEN_GCAL_NOTIFICATIONEMAILER_API_SECRETEMAILER_USER_TAG_ID
Google is disabled unless both GOOGLE_CLIENT_ID and GOOGLE_CLIENT_SECRET are set to real, non-placeholder values. When Google is enabled and the effective Google webhook URL uses HTTPS, TOKEN_GCAL_NOTIFICATION is required for Google Calendar webhook validation.
Derived backend values:
DBis not supplied directly; backend derives it fromNODE_ENVORIGINS_ALLOWEDis derived by splitting the comma-separatedCORSenv var
CLI And Build URL Variables
Primary files:
packages/scripts/src/common/cli.constants.tspackages/scripts/src/common/cli.utils.tspackages/web/build.tspackages/web/dev.ts
Variables used by CLI/build flows:
BASEURL(used for local CLI operations and injected into the web build asAPI_BASEURL)FRONTEND_URL(used by backend auth email flows and CLI domain resolution)
If FRONTEND_URL points to localhost, the CLI prompts for a VM/public domain and builds the API URL from that input.
Web Environment Contract
Source:
packages/web/src/common/constants/env.constants.ts
Important variables:
API_BASEURLNODE_ENVPOSTHOG_KEYPOSTHOG_HOST
GOOGLE_CLIENT_ID is optional. When it is missing, the web app hides Google sign-in and Google Calendar connection actions.
BACKEND_BASEURL is derived from API_BASEURL.
Bun build/dev behavior:
packages/web/build.tsinjectsBASEURLasAPI_BASEURLfor production buildspackages/web/dev.tsdoes the same for the local web dev serverGOOGLE_CLIENT_IDis injected when present- if
API_BASEURLis not injected, the web app falls back toPORTand buildshttp://localhost:<PORT>/api
Practical Mode Matrix
Safe without backend
- route changes
- component rendering
- keyboard and pointer interactions
- local storage behavior
- many task workflows
Requires backend
- user profile loading
- authenticated event APIs
- Google connection flows
- backend validation behavior
- Mongo persistence
- SSE stream behavior
Backend Health Probe
When debugging backend startup or connectivity issues, use the health endpoint first:
curl -i http://localhost:<PORT>/api/health
Interpretation:
200with{"status":"ok","timestamp":"..."}: backend is running and can reach MongoDB500with{"status":"error","timestamp":"..."}: backend is running but database connectivity failed- connection refused/timeouts: backend process is not listening yet, or the port/base URL is wrong
Requires Google-related setup
- real OAuth
- real Google Calendar import/sync
- notification watch flows
Auth And Anonymous Behavior
Compass supports both:
- never-authenticated users using local storage
- authenticated or previously-authenticated users using remote repositories
When testing changes around event loading, explicitly decide which user state you are modeling.
Google Calendar Webhook Notes
Google OAuth and Google Calendar Watch have different local requirements. Google sign-in can use localhost redirect URLs, but Calendar Watch notifications are server-to-server callbacks from Google to Compass. For those callbacks, Google needs an HTTPS backend URL that it can reach from the public internet.
Compass does not start a local tunnel automatically. Google Calendar webhook watch flows use GCAL_WEBHOOK_BASEURL when it is set and fall back to BASEURL when it is not set.
For normal local development:
BASEURL=http://localhost:3000/api
Google sign-in, Google Calendar connect, and initial import can still work, but live Google-to-Compass notifications are skipped because Google cannot call a local HTTP backend.
For local end-to-end Google Watch testing, run a temporary HTTPS tunnel to the backend:
cloudflared tunnel --url http://localhost:3000
Then set:
BASEURL=http://localhost:3000/api
GCAL_WEBHOOK_BASEURL=https://<generated-host>.trycloudflare.com/api
Keep BASEURL local so the browser and Server-Sent Events continue using localhost. Only Google's webhook POST requests should use the tunnel.
Stop the tunnel when testing is complete. Do not use personal calendars with sensitive data for manual tunnel tests.
Common Failure Modes
- backend exits immediately because required env is missing
- backend/web/cli read from
.env.local; using.envinstead leaves required variables unset - web points at the wrong API base URL
- session exists but user profile fetch fails
- sync endpoints work but notification/watch setup is skipped because neither
GCAL_WEBHOOK_BASEURLnorBASEURLis public HTTPS GCAL_WEBHOOK_BASEURLpoints to a tunnel without/api, so Google posts to the wrong route- backend starts but
/api/healthreturns500becauseMONGO_URIor database reachability is broken