# syntax=docker/dockerfile:1 # ─── Stage 1: build SPA + prerender via puppeteer ────────────────────────── FROM node:22-alpine AS builder WORKDIR /app COPY package.json package-lock.json* ./ RUN npm install --no-audit --no-fund RUN apk add --no-cache chromium nss freetype harfbuzz ca-certificates ttf-freefont ENV PUPPETEER_SKIP_DOWNLOAD=true \ PUPPETEER_EXECUTABLE_PATH=/usr/bin/chromium-browser COPY . . RUN npm run build:prerender # ─── Stage 2: runtime (Express) ───────────────────────────────────────────── FROM node:22-alpine WORKDIR /app ENV NODE_ENV=production \ PORT=3000 COPY package.json package-lock.json* ./ RUN npm install --omit=dev --no-audit --no-fund && npm cache clean --force COPY --from=builder /app/dist ./dist COPY --from=builder /app/server ./server COPY --from=builder /app/scripts/pull-external-rss.js ./scripts/pull-external-rss.js COPY --from=builder /app/src/content/feeds.json ./src/content/feeds.json RUN mkdir -p /app/public/uploads /app/data && chown -R node:node /app/public /app/data EXPOSE 3000 HEALTHCHECK --interval=30s --timeout=5s --start-period=15s --retries=3 \ CMD wget --quiet --tries=1 --spider http://127.0.0.1:3000/api/health || exit 1 USER node CMD ["node", "server/index.js"]