Files
yolkbook/docker-compose.yml
derekc aa12648228 Add multi-user auth, admin panel, and timezone support; rename to Yolkbook
- Rename app from Eggtracker to Yolkbook throughout
- Add JWT-based authentication (python-jose, passlib/bcrypt)
- Add users table; all data tables gain user_id FK for full data isolation
- Super admin credentials sourced from ADMIN_USERNAME/ADMIN_PASSWORD env vars,
  synced on every startup; orphaned rows auto-assigned to admin post-migration
- Login page with self-registration; JWT stored in localStorage (30-day expiry)
- Admin panel (/admin): list users, reset passwords, disable/enable, delete,
  and impersonate (Login As) with Return to Admin banner
- Settings modal (gear icon in nav): timezone selector and change password
- Timezone stored per-user; stats date windows computed in user's timezone;
  date input setToday() respects user timezone via Intl API
- migrate_v2.sql for existing single-user installs
- Auto-migration adds timezone column to users on startup
- Updated README with full setup, auth, admin, and migration docs

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-03-17 23:19:29 -07:00

63 lines
2.6 KiB
YAML

services:
# ── MySQL ────────────────────────────────────────────────────────────────────
db:
image: mysql:8.0
restart: unless-stopped
env_file: .env
environment:
MYSQL_ROOT_PASSWORD: ${MYSQL_ROOT_PASSWORD}
MYSQL_DATABASE: ${MYSQL_DATABASE}
MYSQL_USER: ${MYSQL_USER}
MYSQL_PASSWORD: ${MYSQL_PASSWORD}
volumes:
- mysql_data:/var/lib/mysql # persistent data
- ./mysql/init.sql:/docker-entrypoint-initdb.d/init.sql:ro # run once on first start
healthcheck:
test: ["CMD", "mysqladmin", "ping", "-h", "localhost", "-u", "root", "-p${MYSQL_ROOT_PASSWORD}"]
interval: 10s
timeout: 5s
retries: 5
start_period: 30s
networks:
- backend
# ── FastAPI ──────────────────────────────────────────────────────────────────
api:
build: ./backend
restart: unless-stopped
env_file: .env
environment:
DATABASE_URL: mysql+pymysql://${MYSQL_USER}:${MYSQL_PASSWORD}@db/${MYSQL_DATABASE}
ADMIN_USERNAME: ${ADMIN_USERNAME}
ADMIN_PASSWORD: ${ADMIN_PASSWORD}
JWT_SECRET: ${JWT_SECRET}
depends_on:
db:
condition: service_healthy # wait for MySQL to be ready before starting
networks:
- backend
# ── Nginx ────────────────────────────────────────────────────────────────────
nginx:
image: nginx:alpine
restart: unless-stopped
ports:
- "8056:80"
volumes:
- ./nginx/html:/usr/share/nginx/html:ro # static files — edit locally, live immediately
- ./nginx/nginx.conf:/etc/nginx/nginx.conf:ro # nginx config
depends_on:
- api
networks:
- backend
# ── Volumes ───────────────────────────────────────────────────────────────────
volumes:
mysql_data: # survives container restarts and rebuilds
# ── Networks ──────────────────────────────────────────────────────────────────
networks:
backend:
driver: bridge