File size: 2,434 Bytes
6db5454
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
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
# syntax=docker/dockerfile:1.7

# ─── Stage 1: deps ─────────────────────────────────────────────────
# Install only production dependencies. node_modules cached aggressively.
FROM node:20-alpine AS deps
RUN apk add --no-cache libc6-compat
WORKDIR /app

# Lockfile-first copy enables Docker layer caching. Re-runs only on
# package.json / lockfile changes, not on source edits.
COPY package.json package-lock.json* ./
RUN npm ci --no-audit --no-fund


# ─── Stage 2: builder ──────────────────────────────────────────────
# Compile Next.js with standalone output. Outputs go to .next/standalone
# and .next/static β€” both copied into the runtime stage.
FROM node:20-alpine AS builder
WORKDIR /app

COPY --from=deps /app/node_modules ./node_modules
COPY . .

# Build-time env: NEXT_TELEMETRY_DISABLED suppresses telemetry; no secrets here.
# HF_TOKEN is intentionally NOT baked in β€” it's passed at runtime.
ENV NEXT_TELEMETRY_DISABLED=1
RUN npm run build


# ─── Stage 3: runner ───────────────────────────────────────────────
# Minimal runtime image. Only contains the compiled standalone server,
# static assets, and node binary.
FROM node:20-alpine AS runner
WORKDIR /app

ENV NODE_ENV=production
ENV NEXT_TELEMETRY_DISABLED=1
# Default to 7860 (HF Spaces convention). Local docker-compose overrides to 3000.
ENV PORT=7860
ENV HOSTNAME=0.0.0.0

# Run as non-root for defense-in-depth (HF Spaces injects USER 1000 too).
RUN addgroup --system --gid 1001 nextjs \
 && adduser --system --uid 1001 --ingroup nextjs nextjs

# Standalone bundle: server.js + minimum required node_modules.
COPY --from=builder --chown=nextjs:nextjs /app/.next/standalone ./
COPY --from=builder --chown=nextjs:nextjs /app/.next/static ./.next/static
COPY --from=builder --chown=nextjs:nextjs /app/public ./public

USER nextjs
EXPOSE 7860

# Healthcheck β€” uses ash builtin to expand $PORT.
HEALTHCHECK --interval=30s --timeout=5s --start-period=15s --retries=3 \
  CMD wget --quiet --tries=1 --spider "http://localhost:${PORT:-7860}" || exit 1

# server.js (generated by next build with standalone output) reads PORT env var.
CMD ["node", "server.js"]