Files
goodgo-platform/apps/api/Dockerfile
Ho Ngoc Hai b84dfd5cad
Some checks failed
CI / Lint → Typecheck → Test → Build (22) (push) Failing after 11s
CI / E2E Tests (push) Has been skipped
Deploy / Build API Image (push) Failing after 23s
Deploy / Build Web Image (push) Failing after 12s
Deploy / Build AI Services Image (push) Failing after 10s
E2E Tests / Playwright E2E (push) Failing after 12s
Deploy / Deploy to Staging (push) Has been skipped
Deploy / Smoke Test Staging (push) Has been skipped
Deploy / Deploy to Production (push) Has been skipped
Deploy / Smoke Test Production (push) Has been skipped
Deploy / Rollback Staging (push) Has been skipped
Deploy / Rollback Production (push) Has been skipped
fix: Docker build errors — Prisma generate order, .dockerignore multi-service
- Dockerfile: move prisma generate BEFORE nest build (fixes TS2305 PropertyType)
- .dockerignore: remove apps/web + libs/ai-services exclusions (needed by Kaniko)
- CI: add pnpm db:generate step before lint/typecheck/build

Co-Authored-By: Claude Opus 4 (1M context) <noreply@anthropic.com>
2026-04-13 15:31:08 +07:00

82 lines
3.2 KiB
Docker

# =============================================================================
# GoodGo API — Multi-stage Production Dockerfile
# Build: docker build -f apps/api/Dockerfile . (from repo root)
# =============================================================================
# ---- Base ----
FROM node:22-slim AS base
RUN corepack enable && corepack prepare pnpm@10.27.0 --activate
WORKDIR /app
# ---- Dependencies ----
# Install ALL deps (dev + prod) needed for the build step.
# Layer caching: lockfile + manifests change less often than source code.
FROM base AS deps
COPY pnpm-lock.yaml pnpm-workspace.yaml package.json turbo.json ./
COPY apps/api/package.json apps/api/
COPY libs/mcp-servers/package.json libs/mcp-servers/
COPY prisma/ prisma/
RUN pnpm install --frozen-lockfile --filter @goodgo/api...
# ---- Build ----
# Generate Prisma client first (TS types needed at compile time),
# then compile mcp-servers lib (workspace dep), then the NestJS API.
FROM base AS build
COPY --from=deps /app/node_modules ./node_modules
COPY --from=deps /app/apps/api/node_modules ./apps/api/node_modules
COPY --from=deps /app/libs/mcp-servers/node_modules ./libs/mcp-servers/node_modules
COPY tsconfig.base.json ./
COPY prisma/ prisma/
COPY libs/mcp-servers/ libs/mcp-servers/
COPY apps/api/ apps/api/
RUN npx prisma generate \
&& (pnpm --filter @goodgo/mcp-servers build 2>/dev/null || true) \
&& cd apps/api && npx nest build
# Use pnpm deploy to produce a flat, production-only node_modules
# This strips devDependencies and hoists only what @goodgo/api needs.
RUN pnpm deploy --filter @goodgo/api --prod /app/pruned
# ---- Production ----
FROM node:22-slim AS production
LABEL org.opencontainers.image.title="goodgo-api" \
org.opencontainers.image.description="GoodGo Platform NestJS API" \
org.opencontainers.image.vendor="GoodGo" \
org.opencontainers.image.source="https://github.com/goodgo/goodgo-platform-ai"
# dumb-init for proper PID 1 signal handling
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
# Copy pruned production node_modules from pnpm deploy
COPY --from=build --chown=node:node /app/pruned/node_modules ./node_modules
# Copy compiled application
COPY --from=build --chown=node:node /app/apps/api/dist ./dist
# Prisma schema + migrations (needed for runtime client & migrate deploy)
COPY --from=build --chown=node:node /app/prisma ./prisma
# Copy generated Prisma client into node_modules
COPY --from=build --chown=node:node /app/node_modules/.prisma ./node_modules/.prisma
COPY --from=build --chown=node:node /app/node_modules/@prisma/client ./node_modules/@prisma/client
# Package metadata
COPY --from=build --chown=node:node /app/apps/api/package.json ./package.json
# Entrypoint script
COPY --chown=node:node apps/api/docker-entrypoint.sh /usr/local/bin/docker-entrypoint.sh
RUN chmod +x /usr/local/bin/docker-entrypoint.sh
EXPOSE 3001
HEALTHCHECK --interval=30s --timeout=5s --start-period=15s --retries=3 \
CMD node -e "fetch('http://localhost:3001/health').then(r => { if (!r.ok) throw 1 }).catch(() => process.exit(1))"
USER node
ENTRYPOINT ["dumb-init", "--", "docker-entrypoint.sh"]
CMD ["node", "dist/main"]