最近在研究使用Dockerfile制作NodeJs项目的镜像,发现使用官方提供的NodeJs 8.9.4的标准镜像会使得整个镜像的大小达到900M左右,这显然无法接受,所以对较小镜像大小做了几次尝试。
原始Dockerfile如下
FROM node:8.9.4
COPY . /app
WORKDIR /app
RUN npm install pm2 -g
RUN npm install --registry=https://registry.npm.taobao.org
EXPOSE 3000
CMD pm2 /app/pm2/development/processes.json --update-env
使用alpine版本镜像
从Docker Hub镜像仓库中发现主要有三种版本
- node:<version>:This is the defacto image. If you are unsure about what your needs are, you probably want to use this one.(350M)
- node:<version>-slim : This image does not contain the common packages contained in the default tag and only contains the minimal packages needed to run node.(55M)
- node:<version>-alpine : This image is based on the popular Alpine Linux project, available in the
alpine
official image. Alpine Linux is much smaller than most distribution base images (~5MB), and thus leads to much slimmer images in general.(19M)
从上面三个版本可以看出node:<version>-alpine的基础镜像大小只有19M左右,所以我就想是不是可以直接使用该版本作为基础镜像,于是Dockerfile变为:
FROM node:8-alpine
COPY . /app
WORKDIR /app
RUN cnpm install pm2 -g
RUN npm install --registry=https://registry.npm.taobao.org
EXPOSE 3000
CMD pm2 /app/pm2/development/processes.json --update-env
但是我们在build镜像的时候遇到了错误,如下图:
由于我们的项目依赖的包里面需要node-gyp进行rebuild,且需要python支持,而node:8-alpine镜像并不包含python,导致install 无法进行。而如果在该镜像向安装python,那就需要使用python的源码进行编译,编译python源码,又需要安装gcc make等软件包,这一通操作下来,镜像又在不段变大,所以此方案被否决。
多阶段制作镜像
Dockerfile如下:
FROM node:8.9.4 as build
WORKDIR /app
COPY . ./
RUN npm install
FROM node:8-alpine
COPY --from=build /app /app
WORKDIR /app
RUN npm install pm2 -g
EXPOSE 4000
CMD ["pm2-runtime","start","/app/pm2/dev/processes.json"]
第一步基于node 8.9.4制作镜像名称为build,并在其中运行npm install安装依赖。由于使用的node:<version>的镜像,所以不需要我们自己安装一些其他的软件包。
第二步将第一步中包含完整依赖的目录整个copy到第二步制作的镜像中。就是Dockerfile中的这句:COPY --from=build /app /app·通过这种方式我们成功将镜像的大小降低到了150M左右,相较于原来的950M,减少了800M。
所以最后就采用多阶段制作镜像的方式来降低镜像的大小。