Some checks failed
CI / Lint, Typecheck & Test (push) Successful in 2m5s
Deploy — Staging / Lint, Typecheck & Test (push) Successful in 2m5s
Deploy — Staging / Build & push — admin (push) Failing after 3m11s
Deploy — Staging / Build & push — storefront (push) Failing after 2m23s
Deploy — Staging / Deploy to staging VPS (push) Has been skipped
Two root causes for the Docker build failures: 1. convex/_generated/api not found (both apps) turbo prune only traces npm workspace packages; the root convex/ directory is not a workspace package so it is excluded from out/full/. Copy it manually into the prune output after turbo prune runs. 2. @heroui/react not found (storefront) package-lock.json was generated with npm@11 but node:20-alpine ships npm@10. turbo warns it cannot parse the npm 11 lockfile and generates an incomplete out/package-lock.json, causing npm ci inside Docker to miss packages. Upgrade npm to 11 in the deps stage of both Dockerfiles. Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
66 lines
2.7 KiB
Docker
66 lines
2.7 KiB
Docker
# Build context: ./out (turbo prune storefront --docker)
|
|
# out/json/ — package.json files only → used by deps stage for layer caching
|
|
# out/full/ — full pruned monorepo → used by builder stage for source
|
|
# out/package-lock.json
|
|
|
|
# ── Stage 1: deps ────────────────────────────────────────────────────────────
|
|
# Install ALL dependencies (dev + prod) using only the package.json tree.
|
|
# This layer is shared with the builder stage and only rebuilds when
|
|
# a package.json or the lock file changes — not when source code changes.
|
|
FROM node:20-alpine AS deps
|
|
|
|
RUN apk add --no-cache libc6-compat
|
|
WORKDIR /app
|
|
|
|
# Upgrade npm to match the project's packageManager (npm@11). The package-lock.json
|
|
# was generated with npm 11 — npm 10 (bundled with node:20) can't fully parse it,
|
|
# causing turbo prune to generate an incomplete pruned lockfile and npm ci to miss
|
|
# packages like @heroui/react.
|
|
RUN npm install -g npm@11 --quiet
|
|
|
|
COPY json/ .
|
|
COPY package-lock.json .
|
|
RUN npm ci
|
|
|
|
# ── Stage 2: builder ─────────────────────────────────────────────────────────
|
|
# Full monorepo source + build artifact.
|
|
# next build produces .next/standalone/ because output: "standalone" is set
|
|
# in next.config.js — that's what makes the runner stage small.
|
|
FROM node:20-alpine AS builder
|
|
|
|
WORKDIR /app
|
|
|
|
COPY --from=deps /app/node_modules ./node_modules
|
|
COPY full/ .
|
|
|
|
ENV NEXT_TELEMETRY_DISABLED=1
|
|
|
|
RUN npx turbo build --filter=storefront
|
|
|
|
# ── Stage 3: runner ──────────────────────────────────────────────────────────
|
|
# Minimal runtime image — only the standalone bundle, static assets, and public dir.
|
|
# No source code, no dev dependencies, no build tools.
|
|
FROM node:20-alpine AS runner
|
|
|
|
WORKDIR /app
|
|
|
|
ENV NODE_ENV=production \
|
|
NEXT_TELEMETRY_DISABLED=1 \
|
|
HOSTNAME=0.0.0.0 \
|
|
PORT=3000
|
|
|
|
# Non-root user for security
|
|
RUN addgroup -g 1001 -S nodejs && adduser -S nextjs -u 1001
|
|
|
|
# standalone output mirrors the monorepo tree, so server.js lands at /app/server.js
|
|
# Static files and public/ must be copied separately — they are not in standalone/
|
|
COPY --from=builder --chown=nextjs:nodejs /app/apps/storefront/.next/standalone ./
|
|
COPY --from=builder --chown=nextjs:nodejs /app/apps/storefront/.next/static ./apps/storefront/.next/static
|
|
COPY --from=builder --chown=nextjs:nodejs /app/apps/storefront/public ./apps/storefront/public
|
|
|
|
USER nextjs
|
|
|
|
EXPOSE 3000
|
|
|
|
CMD ["node", "server.js"]
|