一. 概述
实验室每次项目发布测试时,都要手动本地打包好了然后上传到服务器,替换原来nginx下面的目录文件,十分麻烦和繁琐。这次就来优化一下,通过Dockerfile + Jenkins实现自动化部署
二. 实践
Nginx相关
如果不了解Nginx的同学,可以看一下我之前的实践,了解一下只用Nginx是怎么实现部署的,这样看后面比较好接受
- 在一个裸的Ubuntu server中我做了哪些 – 掘金 (juejin.cn)
- nginx部署前端页面 – 掘金 (juejin.cn)
- nginx部署前后端分离项目(二) – 掘金 (juejin.cn)
和本篇文章搭配食用,味道更佳哦!!!
Docker
实践
Nginx手动部署需要我们本地执行npm run build
生成打包好的文件上传到服务器,编写Nginx配置文件。
下面以一个常用的Nginx配置文件为例
server{
# 监听本地的3000端口
listen 3000 default_server;
listen [::]:3000 default_server;
# 对应的打包文件目录
root /usr/share/nginx/City;
# history路由需要配置,不然刷新会404
try_files $uri $uri/ /index.html;
# 反向代理,解决跨域
location /api/ {
proxy_pass http://xx.xx.xxx.x:[port]/;
}
}
在项目中需要实现nginx + docker部署,需要我们在工程目录下新建两个文件:Dockerfile
和 default.conf
Dockerfile
是Docker根据Dockerfile生成Docker image用的,default.conf
这个文件名字可以随便取,也可以叫nginx.conf
,不过你需要和你下面编写的Dockerfile配置文件保持一致
接下来我们一起尝试编写一下这两个文件
default.conf
和上面一样,就不详细说了
server{
listen 3000 default_server;
listen [::]:3000 default_server;
root /usr/share/nginx/City;
try_files $uri $uri/ /index.html;
location /api/ {
proxy_pass http://47.92.112.6:8022/;
}
}
Dockerfile
然后我们来编写一个脚本文件,模拟我们打包上传的过程
# 第一阶段
# 使用alpine镜像,可以减少构建后docker镜像文件的体积
FROM node:14-alpine
# 设置工作目录
WORKDIR /project
# 先copy package.json文件到工作目录,一般我们的项目中依赖是不会变的,这样可以充分利用缓存减少部署时的构建时间
COPY package*.json /project/
# 安装node_modules
RUN npm install
# 将所有文件copy到工作目录
COPY . /project
# 开始打包
RUN npm run build
# 第二阶段
# 拉取nginx镜像文件
FROM nginx
# 这里的dist文件就是打包好的文件,project是我们上面设置的工作目录
COPY --from=0 /project/dist /usr/share/nginx/City
# default.conf就是我们项目下面的nginx配置文件,我们需要copy到nginx的相应目录
COPY --from=0 /project/default.conf /etc/nginx/conf.d/City.conf
编写好这两个文件之后,我们只需要上传这个我们这个项目就好了,不需要单独打包上传。这里我将我的文件放到了这个目录下面
然后我们就可以根据Dockerfile生成docker image,再通过image 生成container,直接运行container做端口映射就可以了
# 镜像名设为front-project
docker build -t front-project .
# 运行容器,将本地的8081端口映射到docker的3000端口
docker run -itd -p 8081:3000 --name frontend-container front-project
然后我们直接访问ip+port就可以看到效果啦
注意:映射的3000端口可不能随便乱写哦,这是根据你的Nginx配置文件来得,我上面的default.conf
监听的就是3000端口,所以写的3000
基础知识点
如果你看上面有点懵的话,不如先来了解一下Docker的基础知识点
基本命令
查看所有的容器
docker ps -a
删除容器
docker rm [containerID]
进入容器
docker exec -it [containerId] /bin/bash
生成镜像文件
docker build -t [image-name] .
启动容器
docker run -itd -p 8081:80 --name [containerId] [image]
删除镜像前,要停掉所有与镜像相关的容器
镜像迁移相关
将已有的镜像保存为新的镜像
docker commit mynginx mynginx_i
将镜像 mynginx_i 镜像保存为 mynginx.tar 文件
docker save -o mynginx.tar mynginx_i
将镜像保存为 .tar 文件后就可以将它放到其他服务器进行部署了,首先将其恢复;
docker load -i mynginx.tar
docker images
Dockerfile多阶段构建
Dockerfile 文件,其内包含了一条条的指令,每一条指令构建一层,因此每一条指令的内容,就是描述该层如何构建
Docker镜像并不是由一个文件构成,而是由一堆文件构成,最主要的文件就是层,每一层构建完就不会再发生改变,后一层的改变只发生在自己的那一层,镜像层会被缓存和复用
镜像是由多层文件系统组成,想要优化它的大小,就需要去减少层数、每一层尽量只包含该层需要的东西,任何额外的东西应该在该层构建结束前清理掉,下面开始正文
镜像上传到Docker Hub, 体积小拉的化就会很快,相当于一个docker构建部署的一个优化
镜像优化
- 使用node的Alpine版本,减少镜像文件体积
- 减少层数,不经常变动的层移到前面去
- env设置多个环境变量,我们只需要dependencies的依赖,而devDependencies 依赖只是编译阶段用的,比如eslint,ts,也就是说最终镜像只需要我们需要的东西
总结:镜像构建的几个阶段
我把这个构建分成了三个阶段:
-
第一阶段:构建基础镜像
安装依赖、编译、运行等等阶段,就是所有阶段共用的东西都在第一阶段封到一个基础镜像里供其它阶段使用,比如设置环境变量、设置工作目录、安装 nodejs、yarn 等等
-
第二阶段:安装依赖阶段
在这个阶段,安装依赖,如果项目需要编译,可以在这个阶段安装依赖编译好
npm install
命令安装依赖比较耗时,大部分时我们只是改变代码不改变依赖,package.json可以缓存起来提高编译速度这里再说下装依赖的小细节,就是执行
yarn --production
加个 production 参数或者环境变量NODE_ENV
为production
,yarn 将不会安装 devDependencies 中列出的任何软件包,点我查看官方文档说明 -
第三阶段:最终使用镜像
拷贝第二阶段安装的好的依赖文件夹,然后在拷贝代码文件到工作目录,执行启动命令,第二阶段装依赖多出的一些垃圾我们不需要,我们就只拷贝我们要用的东西,大大减少镜像的大小
如果项目需要编译,在拷贝编译后的文件夹,不需要拷贝编译前的代码,有编译后的代码和依赖就可以跑起项目
多阶段构建,最后生成的镜像只能是最后一个阶段的结果,但是,能够将前置阶段中的文件拷贝到后边的阶段中,这就是多阶段构建的最大意义。
通过多阶段构建,充分利用缓存,可以将二次构建的速度提升到3.6秒
jenkins
了解完Docker之后,我们就来学习一下怎么将这两者结合,最终实现自动部署
安装jenkins
jenkins安装也挺麻烦的,还涉及到Java版本等其他问题,这里我也是直接拉取jenkins镜像来安装jenkins
镜像文件地址:hub.docker.com/r/jenkinsci…
# 拉取镜像
docker pull jenkinsci/blueocean
根据镜像运行容器,并进行数据映射
因为之后我们需要在jenkins中操作docker容器,生成镜像文件并运行容器,所以要做一下数据映射
在jenkins中执行docker有三种方式:
- Jenkins中安装docker客户端,使用第三方的docker 需要付费
- Jenkins中安装docker客户端,自己搭建docker,也就是 docker in docker, 比较麻烦,需要特权模式,或者使用第三方工具,比较复杂
- Jenkins中什么都不装,直接使用宿主机的docker服务
这里我们就是采用的第三种方案:
-
优势:简单,方便,直观
-
缺点:jenkins直接使用宿主机的docker服务,Jenkins可以全权的管理所有的容器(包括Jenkins自己),有隐含的安全问题
# 运行容器
docker run -itd \
-u root \
--name jenkins_auto \
-p 5002:8080 \
-v /root/docker/jenkins:/var/jenkins_home \
-v /var/run/docker.sock:/var/run/docker.sock \
jenkinsci/blueocean
然后我们进去jenkins容器里面,看看是不是能操作docker
# 进入容器
docker exec -it afbd2729c2af bash
# 是否有docker命令
docker --help
我们输一下docker images
,发现确实可以操作宿主机上的所有容器
运行Jenkins
第一次登陆需要密码,因为我们做了数据映射,所以直接到我们映射过的目录下面找就好了
cat root/docker/jenkins/secrets/initiaLAdminPassword
然后跟着引导创建一个新的用户就好,安装推荐的创建,失败了也不要紧,到时候需要啥插件我们自己装就好
安装插件
因为jenkins版本和插件版本的原因,安装了好几次插件都报错,最后选择了直接下载到本地然后上传插件的方式
插件下载地址:Index of /download/plugins (jenkins-ci.org)
安装git插件
点击插件管理 -> 高级
上传我们之前下载到本地的git插件
新建任务
点击配置
输入我们的仓库地址和账号密码(点击添加,可以新增一个账号)
构建的时候选择执行shell
输入我们的shell脚本
# 先删除之前的容器和镜像文件
if [ "$(docker ps -a | grep jenkins-front-city)" ]; then
docker stop jenkins-front-city
docker rm jenkins-front-city
fi
if [ "$(docker images -q jenkins-front-city)" ]; then
docker rmi jenkins-front-city
fi
# 重新生成
docker build -t jenkins-front-city .
docker run -itd -p 8866:3000 --name jenkins-front-city jenkins-front-city
最后我们点击保存就大功告成了!!!点击构建之后就能看到最后效果了!!!
结尾
各位看官如果觉得有点收获的话,就动动手指给个免费的赞吧!
Ps:
别人的520:有人相爱,有人夜里开车看海
我的520:比较构建速度
愿历经千辛,归来还是你
往期推荐:
原文链接:https://juejin.cn/post/7235091963312717885 作者:晨出