Dockerfile 1.7 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162
  1. # Build frontend
  2. FROM node:22-bookworm-slim AS frontend-builder
  3. WORKDIR /app/frontend
  4. # Copy package files first for better caching
  5. COPY frontend/package*.json ./
  6. # Use cache mount for npm
  7. RUN --mount=type=cache,target=/root/.npm \
  8. npm ci
  9. COPY frontend/ ./
  10. RUN npm run build
  11. # Production image
  12. FROM python:3.13-slim
  13. WORKDIR /app
  14. # Install system dependencies
  15. ENV DEBIAN_FRONTEND=noninteractive
  16. RUN apt-get update && apt-get install -y --no-install-recommends \
  17. curl \
  18. ffmpeg \
  19. && rm -rf /var/lib/apt/lists/*
  20. # Install Python dependencies with cache mount
  21. COPY requirements.txt ./
  22. RUN --mount=type=cache,target=/root/.cache/pip \
  23. pip install --root-user-action=ignore -r requirements.txt
  24. # Copy backend
  25. COPY backend/ ./backend/
  26. # Copy built frontend from builder stage
  27. COPY --from=frontend-builder /app/static ./static
  28. # Create data directory for persistent storage
  29. # chmod 777 allows running as non-root user (e.g., with docker compose user: directive)
  30. RUN mkdir -p /app/data /app/logs && chmod 777 /app/data /app/logs
  31. # Environment variables
  32. ENV PYTHONUNBUFFERED=1
  33. ENV DATA_DIR=/app/data
  34. ENV LOG_DIR=/app/logs
  35. ENV PORT=8000
  36. EXPOSE 3000
  37. EXPOSE 8000
  38. EXPOSE 8883
  39. EXPOSE 9990
  40. EXPOSE 50000-50100
  41. # Health check (uses PORT env var via shell)
  42. HEALTHCHECK --interval=30s --timeout=10s --start-period=10s --retries=3 \
  43. CMD python -c "import urllib.request, os; urllib.request.urlopen(f'http://localhost:{os.environ.get(\"PORT\", \"8000\")}/health')" || exit 1
  44. # Run the application
  45. # Use standard asyncio loop (uvloop has permission issues in some Docker environments)
  46. # Port is configurable via PORT environment variable (default: 8000)
  47. CMD ["sh", "-c", "uvicorn backend.app.main:app --host 0.0.0.0 --port ${PORT:-8000} --loop asyncio"]