Dockerfile 2.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107
  1. # syntax=docker/dockerfile:1
  2. ARG OPT_DIR="/opt"
  3. ARG PNPM_HOME="/root/.local/share/pnpm"
  4. ##
  5. ## base — DHI dev image with pnpm + turbo
  6. ##
  7. FROM dhi.io/node:24-debian13-dev AS base
  8. ARG OPT_DIR
  9. ARG PNPM_HOME
  10. WORKDIR $OPT_DIR
  11. # Install build dependencies
  12. RUN --mount=type=cache,target=/var/lib/apt,sharing=locked \
  13. --mount=type=cache,target=/var/cache/apt,sharing=locked \
  14. apt-get update && apt-get install -y --no-install-recommends ca-certificates wget
  15. # Install pnpm (standalone script, no version hardcoding)
  16. RUN wget -qO- https://get.pnpm.io/install.sh | ENV="$HOME/.shrc" SHELL="$(which sh)" sh -
  17. ENV PNPM_HOME=$PNPM_HOME
  18. ENV PATH="$PNPM_HOME:$PATH"
  19. # Install turbo globally
  20. RUN --mount=type=cache,target=$PNPM_HOME/store,sharing=locked \
  21. pnpm add turbo --global
  22. ##
  23. ## builder — build + produce artifacts (current 3-stage COPY . . pattern)
  24. ##
  25. FROM base AS builder
  26. ARG OPT_DIR
  27. ARG PNPM_HOME
  28. ENV PNPM_HOME=$PNPM_HOME
  29. ENV PATH="$PNPM_HOME:$PATH"
  30. WORKDIR $OPT_DIR
  31. COPY . .
  32. RUN --mount=type=cache,target=$PNPM_HOME/store,sharing=locked \
  33. pnpm add node-gyp --global
  34. RUN --mount=type=cache,target=$PNPM_HOME/store,sharing=locked \
  35. pnpm install --frozen-lockfile
  36. # Build
  37. RUN turbo run clean
  38. RUN turbo run build --filter @growi/app
  39. # Produce artifacts
  40. RUN pnpm deploy out --prod --filter @growi/app
  41. RUN rm -rf apps/app/node_modules && mv out/node_modules apps/app/node_modules
  42. RUN rm -rf apps/app/.next/cache
  43. RUN tar -zcf /tmp/packages.tar.gz \
  44. package.json \
  45. apps/app/.next \
  46. apps/app/config \
  47. apps/app/dist \
  48. apps/app/public \
  49. apps/app/resource \
  50. apps/app/tmp \
  51. apps/app/.env.production* \
  52. apps/app/next.config.js \
  53. apps/app/package.json \
  54. apps/app/node_modules
  55. ##
  56. ## release — DHI runtime (no shell, no additional binaries)
  57. ##
  58. FROM dhi.io/node:24-debian13 AS release
  59. ARG OPT_DIR
  60. ENV NODE_ENV="production"
  61. ENV appDir="$OPT_DIR/growi"
  62. # Extract artifacts as node user
  63. USER node
  64. WORKDIR ${appDir}
  65. RUN --mount=type=bind,from=builder,source=/tmp/packages.tar.gz,target=/tmp/packages.tar.gz \
  66. tar -zxf /tmp/packages.tar.gz -C ${appDir}/
  67. # Copy TypeScript entrypoint
  68. COPY --chown=node:node apps/app/docker-new/docker-entrypoint.ts /docker-entrypoint.ts
  69. # Switch back to root for entrypoint (it handles privilege drop)
  70. USER root
  71. WORKDIR ${appDir}/apps/app
  72. # OCI standard labels
  73. LABEL org.opencontainers.image.source="https://github.com/weseek/growi"
  74. LABEL org.opencontainers.image.title="GROWI"
  75. LABEL org.opencontainers.image.description="Team collaboration wiki using Markdown"
  76. LABEL org.opencontainers.image.vendor="WESEEK, Inc."
  77. VOLUME /data
  78. EXPOSE 3000
  79. ENTRYPOINT ["node", "/docker-entrypoint.ts"]