1、在 express
项目根目录下新增 Dockerfile
文件,内容如下:
创建服务容器的方法,可以根据自己的情况选择:
1、以下示例为宿主机没有安装node
环境的写法;
2、先在本地构建包含node
和express
的基础镜像,再将构建好的镜像传到服务器直接基于构建好的镜像创建容器;
3、在宿主机安装node
环境,将express
工程源码上传到服务器,通过sh
脚本在服务器构建镜像,再基于构建好的镜像创建容器。
Dockerfile
# 基础镜像为 Alpine Linux 版本 3.13, 使用 Alpine Linux 作为容器的基础操作系统
FROM alpine:3.13# 容器默认时区为UTC,如需使用上海时间请启用以下时区设置命令
# RUN apk add tzdata && cp /usr/share/zoneinfo/Asia/Shanghai /etc/localtime && echo Asia/Shanghai > /etc/timezone# 安装依赖包,如需其他依赖包,请到alpine依赖包管理(https://pkgs.alpinelinux.org/packages?name=php8*imagick*&branch=v3.13)查找。
RUN apk add --update --no-cache nodejs npm# 指定工作目录
WORKDIR /app# 拷贝包管理文件
COPY package*.json /app# npm 源,选用国内镜像源以提高下载速度
RUN npm config set registry https://registry.npm.taobao.org/# npm 安装依赖
RUN npm install# 将当前目录(dockerfile所在目录)下所有文件都拷贝到工作目录下(.gitignore中的文件除外)
COPY . /app# 执行启动命令.
# 写多行独立的CMD命令是错误写法!只有最后一行CMD命令会被执行,之前的都会被忽略,导致业务报错。
# 请参考[Docker官方文档之CMD命令](https://docs.docker.com/engine/reference/builder/#cmd)
# 执行 package.json 的 scripts 中约定的自定义命令时,格式必须为 CMD ["npm", "run", "命令"]
CMD ["npm", "run", "pro"]
2、在 express
项目根目录下新增 deploy.sh
部署脚本,内容如下:
# 停止正在运行的容器
docker stop container_name# 删除正在运行的容器
docker rm -f container_name# 删除已存在的镜像
docker rmi image_name:latest # 构建镜像
docker build -t image_name:latest . # 基于构建的镜像创建容器,并将服务的日志目录挂载到宿主机,没有日志可以不挂,视自己情况挂载
docker run -d -p 3000:3000 -v /path/logs:/app/logs --name container_name image_name
3、在 express
项目根目录下新增 .dockerignore
文件,用于指定在构建 Docker
镜像时需要被忽略的文件和目录,内容如下:
Dockerfile
.dockerignore
node_modules
npm-debug.log
4、将 express
项目文件上传至服务器任意目录,进入项目根目录,执行如下命令,成功后会显示容器的标识符(长字符串)
sh ./deploy.sh
docker images // 查看生成的镜像
docker ps -a // 查看已经创建的所有容器,包括失败的
如果 STATUS
为 UP
则表示成功,失败的话可以执行 docker logs container_name
查看原因
5、配置 nginx
代理,主要内容如下:
没有
ssl
证书则可以不配ssl_
开头的参数
server {listen 80;listen 443 ssl;server_name demo.com;ssl_certificate /etc/nginx/ssl/fullchain.pem;ssl_certificate_key /etc/nginx/ssl/privkey.key;ssl_session_timeout 5m;ssl_protocols TLSv1 TLSv1.1 TLSv1.2;ssl_ciphers ECDHE-RSA-AES128-GCM-SHA256:HIGH:!aNULL:!MD5:!RC4:!DHE;ssl_prefer_server_ciphers on;# listen 80;# listen [::]:80;client_max_body_size 10m;#access_log /var/log/nginx/host.access.log main;location /api/ {proxy_set_header HOST $host;proxy_set_header X-Forwarded-Proto $scheme;proxy_set_header X-Real-IP $remote_addr;proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;# 如果是腾讯云服务器,xxx只能是内网 ip,被卡(keng)了很久proxy_pass http://xxx:3000/; }location / {root /usr/share/nginx/html;index index.html index.htm;}# ...
}
6、重启 nginx
服务后,就能正常使用了,如果是容器部署的 nginx
,则直接重启容器就行
docker restart nginx