# ---- Base ---- FROM node:22-slim AS base RUN corepack enable && corepack prepare pnpm@10.27.0 --activate WORKDIR /app # ---- Dependencies ---- FROM base AS deps COPY pnpm-lock.yaml pnpm-workspace.yaml package.json turbo.json ./ COPY apps/web/package.json apps/web/ RUN pnpm install --frozen-lockfile --filter @goodgo/web... # ---- Build ---- FROM base AS build COPY --from=deps /app/node_modules ./node_modules COPY --from=deps /app/apps/web/node_modules ./apps/web/node_modules COPY tsconfig.base.json ./ COPY apps/web/ apps/web/ RUN cd apps/web && npx next build # ---- Production ---- FROM node:22-slim AS production RUN apt-get update && apt-get install -y --no-install-recommends dumb-init && rm -rf /var/lib/apt/lists/* WORKDIR /app ENV NODE_ENV=production ENV NEXT_TELEMETRY_DISABLED=1 ENV HOSTNAME=0.0.0.0 ENV PORT=3000 # Install production deps fresh (pnpm standalone output has broken symlinks) COPY --from=deps /app/pnpm-lock.yaml /app/pnpm-workspace.yaml /app/package.json /app/turbo.json ./ COPY --from=deps /app/apps/web/package.json ./apps/web/ RUN corepack enable && corepack prepare pnpm@10.27.0 --activate \ && printf '#!/bin/sh\nexit 0' > /usr/local/bin/husky && chmod +x /usr/local/bin/husky \ && pnpm install --frozen-lockfile --filter @goodgo/web... --prod # Copy standalone server.js (the entrypoint Next.js generates) COPY --from=build --chown=node:node /app/apps/web/.next/standalone/apps/web/server.js ./server.js # Copy .next build output COPY --from=build --chown=node:node /app/apps/web/.next/standalone/apps/web/.next ./.next COPY --from=build --chown=node:node /app/apps/web/.next/static ./.next/static # Copy public assets if any exist (may be empty) COPY --from=build --chown=node:node /app/apps/web/public ./public EXPOSE 3000 HEALTHCHECK --interval=30s --timeout=5s --start-period=10s --retries=3 \ CMD node -e "fetch('http://localhost:3000/api/health').then(r => { if (!r.ok) throw 1 }).catch(() => { process.exit(1) })" USER node ENTRYPOINT ["dumb-init", "--"] CMD ["node", "server.js"]