跳至主要内容
版本:9.x

使用 Docker

注意

在构建时,无法在 Docker 容器和主机文件系统之间创建 reflinks 或 hardlinks。您可以做的最好的事情是使用 BuildKit 缓存挂载来在构建之间共享缓存。或者,您可以使用 podman,因为它可以在构建时挂载 Btrfs 卷。

最小化 Docker 镜像大小和构建时间

  • 使用较小的镜像,例如 node:XX-slim
  • 尽可能利用多阶段构建,如果合理。
  • 利用 BuildKit 缓存挂载。

示例 1:在 Docker 容器中构建包

由于 devDependencies 仅在构建包时需要,pnpm install --prod 将与 pnpm installpnpm run build 处于不同的阶段,允许最终阶段仅从早期阶段复制必要的文件,从而最小化最终镜像的大小。

.dockerignore
node_modules
.git
.gitignore
*.md
dist
Dockerfile
FROM node:20-slim AS base
ENV PNPM_HOME="/pnpm"
ENV PATH="$PNPM_HOME:$PATH"
RUN corepack enable
COPY . /app
WORKDIR /app

FROM base AS prod-deps
RUN --mount=type=cache,id=pnpm,target=/pnpm/store pnpm install --prod --frozen-lockfile

FROM base AS build
RUN --mount=type=cache,id=pnpm,target=/pnpm/store pnpm install --frozen-lockfile
RUN pnpm run build

FROM base
COPY --from=prod-deps /app/node_modules /app/node_modules
COPY --from=build /app/dist /app/dist
EXPOSE 8000
CMD [ "pnpm", "start" ]

示例 2:在单仓库中构建多个 Docker 镜像

假设您有一个包含 3 个包的单仓库:app1、app2 和 common;app1 和 app2 依赖于 common,但彼此不依赖。

您希望仅保存每个包所需的依赖项,pnpm deploy 应该可以帮助您仅复制必要的文件和包。

单仓库的结构
./
├── Dockerfile
├── .dockerignore
├── .gitignore
├── packages/
│   ├── app1/
│   │   ├── dist/
│   │   ├── package.json
│   │   ├── src/
│   │   └── tsconfig.json
│   ├── app2/
│   │   ├── dist/
│   │   ├── package.json
│   │   ├── src/
│   │   └── tsconfig.json
│   └── common/
│   ├── dist/
│   ├── package.json
│   ├── src/
│   └── tsconfig.json
├── pnpm-lock.yaml
├── pnpm-workspace.yaml
└── tsconfig.json
pnpm-workspace.yaml
packages:
- 'packages/*'
.dockerignore
node_modules
.git
.gitignore
*.md
dist
Dockerfile
FROM node:20-slim AS base
ENV PNPM_HOME="/pnpm"
ENV PATH="$PNPM_HOME:$PATH"
RUN corepack enable

FROM base AS build
COPY . /usr/src/app
WORKDIR /usr/src/app
RUN --mount=type=cache,id=pnpm,target=/pnpm/store pnpm install --frozen-lockfile
RUN pnpm run -r build
RUN pnpm deploy --filter=app1 --prod /prod/app1
RUN pnpm deploy --filter=app2 --prod /prod/app2

FROM base AS app1
COPY --from=build /prod/app1 /prod/app1
WORKDIR /prod/app1
EXPOSE 8000
CMD [ "pnpm", "start" ]

FROM base AS app2
COPY --from=build /prod/app2 /prod/app2
WORKDIR /prod/app2
EXPOSE 8001
CMD [ "pnpm", "start" ]

运行以下命令以构建 app1 和 app2 的镜像

docker build . --target app1 --tag app1:latest
docker build . --target app2 --tag app2:latest