# Multi-Stage Build: Frontend bauen, dann Backend bauen, dann schlankes Runtime-Image # --------------------------------------------------------------------------------- # ============== STAGE 1: Frontend bauen ============== FROM node:20-alpine AS frontend-builder WORKDIR /build/frontend COPY frontend/package.json frontend/package-lock.json ./ RUN npm ci --no-audit --no-fund --prefer-offline COPY frontend/ ./ RUN npm run build # Output: /build/frontend/dist/ # ============== STAGE 2: Backend bauen (TS → JS) ============== FROM node:20-alpine AS backend-builder WORKDIR /build/backend COPY backend/package.json backend/package-lock.json ./ RUN npm ci --no-audit --no-fund --prefer-offline COPY backend/prisma ./prisma RUN npx prisma generate COPY backend/tsconfig.json ./ COPY backend/src ./src RUN npx tsc # Output: /build/backend/dist/ # ============== STAGE 3: Runtime ============== FROM node:20-alpine WORKDIR /app # Nur Production-Dependencies + Prisma-Client COPY backend/package.json backend/package-lock.json ./ RUN npm ci --omit=dev --no-audit --no-fund --prefer-offline && npm cache clean --force # Build-Artefakte aus Stage 2 COPY --from=backend-builder /build/backend/dist ./dist COPY --from=backend-builder /build/backend/node_modules/.prisma ./node_modules/.prisma COPY --from=backend-builder /build/backend/node_modules/@prisma ./node_modules/@prisma COPY backend/prisma ./prisma # Frontend-Build ins public/-Verzeichnis (wird in production-Mode statisch ausgeliefert) COPY --from=frontend-builder /build/frontend/dist ./public # Daten-Verzeichnisse (werden via Bind-Mount überlagert; hier nur als Fallback) RUN mkdir -p uploads factory-defaults prisma/backups # Healthcheck HEALTHCHECK --interval=30s --timeout=5s --start-period=20s --retries=3 \ CMD wget --quiet --tries=1 --spider "http://localhost:${PORT:-3001}/api/health" || exit 1 # Beim Start: prisma db push (idempotent), dann node COPY backend/docker-entrypoint.sh /usr/local/bin/docker-entrypoint.sh RUN chmod +x /usr/local/bin/docker-entrypoint.sh ENTRYPOINT ["docker-entrypoint.sh"] CMD ["node", "dist/index.js"]