File size: 4,000 Bytes
e327f0d
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
# syntax=docker/dockerfile:1.7
# ---------------------------------------------------------------------------
# Next.js 15 (App Router) β€” pnpm monorepo standalone build
# Build context: monorepo root (services/backend/docker-compose.yml passes ../..)
# ---------------------------------------------------------------------------

ARG NODE_VERSION=20-alpine
ARG PNPM_VERSION=9.12.0

# ===========================================================================
# Stage 1: deps  β€” install workspace dependencies (cache-friendly)
# ===========================================================================
FROM node:${NODE_VERSION} AS deps
ARG PNPM_VERSION
WORKDIR /app

# Alpine essentials for some native modules (sharp, etc.)
RUN apk add --no-cache libc6-compat

# Enable corepack and pin pnpm version (matches root packageManager field)
RUN corepack enable && corepack prepare pnpm@${PNPM_VERSION} --activate

# Copy workspace manifests first for better layer caching
COPY package.json pnpm-lock.yaml pnpm-workspace.yaml ./
COPY apps/web/package.json ./apps/web/
COPY packages/types/package.json ./packages/types/
COPY packages/ui/package.json ./packages/ui/

# Install only the deps needed for the web app + its workspace siblings
RUN --mount=type=cache,id=pnpm,target=/root/.local/share/pnpm/store \
    pnpm install --frozen-lockfile \
      --filter @arac-hasar/web... \
      --filter @arac-hasar/ui \
      --filter @arac-hasar/types

# ===========================================================================
# Stage 2: builder β€” copy source, run next build (produces standalone bundle)
# ===========================================================================
FROM node:${NODE_VERSION} AS builder
ARG PNPM_VERSION
WORKDIR /app

RUN apk add --no-cache libc6-compat
RUN corepack enable && corepack prepare pnpm@${PNPM_VERSION} --activate

# Bring in fully installed node_modules graph from deps stage
COPY --from=deps /app/node_modules ./node_modules
COPY --from=deps /app/apps/web/node_modules ./apps/web/node_modules
COPY --from=deps /app/packages/types/node_modules ./packages/types/node_modules
COPY --from=deps /app/packages/ui/node_modules ./packages/ui/node_modules

# Copy workspace manifests + source
COPY package.json pnpm-lock.yaml pnpm-workspace.yaml tsconfig.base.json ./
COPY apps/web ./apps/web
COPY packages/types ./packages/types
COPY packages/ui ./packages/ui

# Public env is *baked* at build time (Next.js NEXT_PUBLIC_*)
# Default points at host-mapped backend so the browser can reach it.
ARG NEXT_PUBLIC_API_URL=http://localhost:8000
ENV NEXT_PUBLIC_API_URL=${NEXT_PUBLIC_API_URL}
ENV NEXT_TELEMETRY_DISABLED=1
ENV NODE_ENV=production

# Build the web app β€” standalone output lands in apps/web/.next/standalone
RUN pnpm --filter @arac-hasar/web build

# ===========================================================================
# Stage 3: runner β€” minimal production image
# ===========================================================================
FROM node:${NODE_VERSION} AS runner
WORKDIR /app

ENV NODE_ENV=production
ENV NEXT_TELEMETRY_DISABLED=1
ENV PORT=3000
ENV HOSTNAME=0.0.0.0

RUN apk add --no-cache wget \
 && addgroup --system --gid 1001 nodejs \
 && adduser  --system --uid 1001 nextjs

# Copy standalone server (self-contained Node bundle that includes
# the minimal node_modules graph, including workspace packages).
# Standalone places the server.js inside apps/web/ when built from a monorepo.
COPY --from=builder --chown=nextjs:nodejs /app/apps/web/.next/standalone ./
COPY --from=builder --chown=nextjs:nodejs /app/apps/web/.next/static       ./apps/web/.next/static
COPY --from=builder --chown=nextjs:nodejs /app/apps/web/public             ./apps/web/public

USER nextjs

EXPOSE 3000

HEALTHCHECK --interval=30s --timeout=5s --start-period=20s --retries=3 \
  CMD wget -qO- http://localhost:3000/api/health || exit 1

# server.js sits next to the apps/web directory in the standalone output
CMD ["node", "apps/web/server.js"]