feat: add Docker containerization (Dockerfile, compose, dockerignore)

Multi-stage build: frontend (Vite) + backend (TypeScript) in one container.
Production image based on node:20-alpine with health check.

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
This commit is contained in:
Ronny
2026-04-06 10:05:10 +02:00
parent 06ed0ec3ce
commit 7bd22b53e9
3 changed files with 107 additions and 0 deletions
+20
View File
@@ -0,0 +1,20 @@
node_modules
dist
.git
.gitignore
.DS_Store
*.md
*.docx
*.html
.vscode
.claude
.mcp.json
tools.yaml
backend/public
backend/node_modules
backend/dist
backend/.env
frontend/node_modules
frontend/dist
prototyp_*.html
projektplan_*
+53
View File
@@ -0,0 +1,53 @@
# ============================================================
# Stage 1: Build Frontend (React + Vite)
# ============================================================
FROM node:20-alpine AS build-frontend
WORKDIR /app/frontend
COPY frontend/package.json frontend/package-lock.json ./
RUN npm ci
COPY frontend/ ./
RUN npm run build
# Output: /app/frontend/../backend/public → /app/backend/public
# ============================================================
# Stage 2: Build Backend (TypeScript → JavaScript)
# ============================================================
FROM node:20-alpine AS build-backend
WORKDIR /app/backend
COPY backend/package.json backend/package-lock.json ./
RUN npm ci
COPY backend/ ./
RUN npx tsc
# ============================================================
# Stage 3: Production Image
# ============================================================
FROM node:20-alpine AS production
WORKDIR /app
# Nur Production-Dependencies installieren
COPY backend/package.json backend/package-lock.json ./
RUN npm ci --omit=dev && npm cache clean --force
# Compiled Backend
COPY --from=build-backend /app/backend/dist ./dist
# Frontend Build (statische Dateien)
COPY --from=build-frontend /app/backend/public ./public
# Non-root User
RUN addgroup -S appgroup && adduser -S appuser -G appgroup
USER appuser
ENV NODE_ENV=production
ENV PORT=3001
EXPOSE 3001
HEALTHCHECK --interval=30s --timeout=5s --start-period=10s --retries=3 \
CMD wget -qO- http://localhost:3001/health || exit 1
CMD ["node", "dist/index.js"]
+34
View File
@@ -0,0 +1,34 @@
services:
tippspiel:
build:
context: .
dockerfile: Dockerfile
container_name: wm2026-tippspiel
restart: unless-stopped
ports:
- "3301:3001"
environment:
- NODE_ENV=production
- PORT=3001
- DATABASE_URL=${DATABASE_URL}
- SUPABASE_URL=${SUPABASE_URL}
- SUPABASE_SERVICE_ROLE_KEY=${SUPABASE_SERVICE_ROLE_KEY}
- ANTHROPIC_API_KEY=${ANTHROPIC_API_KEY}
- FOOTBALL_API_KEY=${FOOTBALL_API_KEY}
- FOOTBALL_API_BASE_URL=https://api.football-data.org/v4
- ELEVENLABS_API_KEY=${ELEVENLABS_API_KEY}
- CORS_ORIGIN=${CORS_ORIGIN:-https://app.staffbase.com,http://localhost:5173}
- STAFFBASE_PUBLIC_KEY=${STAFFBASE_PUBLIC_KEY:-}
- STAFFBASE_PLUGIN_ID=${STAFFBASE_PLUGIN_ID:-}
healthcheck:
test: ["CMD", "wget", "-qO-", "http://localhost:3001/health"]
interval: 30s
timeout: 5s
start_period: 10s
retries: 3
networks:
- main-network
networks:
main-network:
external: true