从docker到docker-compose部署一个nginx+flask+mysql+redis应用
目的是把一个flask项目的mysql数据库、redis数据库、flask应用、nginx服务分别装到四个容器中,然后用docker-compose命令同时启动与关闭
一、安装docker
Docker 运行在 CentOS 7 上,要求系统为64位、系统内核版本为 3.10 以上。通过 uname -r 命令查看当前的内核版本
更新yum
sudo yum update
移除旧版本
sudo yum remove docker docker-client docker-client-latest docker-common docker-latest docker-latest-logrotate docker-logrotate docker-selinux docker-engine-selinux docker-engine
安装系统工具
sudo yum install -y yum-utils device-mapper-persistent-data lvm2
添加软件源
sudo yum-config-manager --add-repo http://mirrors.aliyun.com/docker-ce/linux/centos/docker-ce.repo
更新yum缓存
sudo yum makecache fast
安装docker-ce,对所有选项选y
sudo yum -y install docker-ce
查看docker版本
docker --version
启动docker
sudo systemctl start docker
二、只用docker部署
注意区分:docker镜像是一个模版,docker容器是一个实例,它可以被启动与关闭。
1、首先先把mysql和redis数据库跑起来
拉镜像
docker pull mysql docker pull redis:3.2
运行,redis并不需要把端口暴露出来,因为只有flask应用需要访问它,直接在启动flask应用容器的时候link到redis就可以了
docker run --name mysql -p 3306:3306 -e MYSQL_ROOT_PASSWORD=*** -d mysql docker run --name redis -d redis:3.2 redis-server --appendonly yes
2、部署flask应用
一开始我直接拉了一个centos镜像,然后再装python3、requirement.txt,但是后来发现直接用python:3.6的镜像会简单一点。
首先,把项目复制到服务器
scp -r 本地文件夹 root@服务器公网ip:服务器目标文件夹
然后,创建一个Dockerfile,用于将应用build成镜像
目录结构如下
Dockerfile的内容
FROM python:3.6 ADD ./my-chat-server /app WORKDIR /app RUN pip install -r requirement.txt CMD ["python", "run.py"]
RUN与CMD的区别在于,RUN是在创建镜像的时候执行的,而CMD在这个镜像创建的容器每次启动时都会执行
ADD与COPY的区别在于,ADD可以写成 ADD http://foo.com/bar.go /tmp/main.go 或
ADD /foo.tar.gz /tmp/ 用于下载一个文件到容器中或者将一个压缩文件解压到容器中。
然后,在flask目录执行
docker build -t flask .
这会创建一个名为flask的镜像
然后启动这个镜像
docker run --name flask -p 5000:5000 --link mysql:mysql --link redis:redis-server -d flask
注意应用中mysql和redis的host都要相应改变,改成数据库容器在flask容器中的别名
--link 容器名:容器别名 可以实现容器间的通信
如无意外现在可以从服务器的5000端口访问到flask应用了
3、部署nginx
先拉一个镜像
docker pull nginx
然后启动
docker run --name nginx -p 80:80 -d nginx
将里面的nginx.conf文件内容拿出来
docker exec -it nginx cat /etc/nginx/nginx.conf
然后创建一个nginx.conf,把这些内容复制进去,在http里加上
server { listen 80; server_name 服务器公网ip; location / { proxy_pass http://flask:5000; } access_log /var/log/access_unicom.log main; }
然后把刚创的nginx镜像删掉,注意删除容器是docker rm;删除镜像是docker rmi
创建一个以刚才创建的nginx.conf为配置的nginx容器
docker run --name nginx -p 80:80 -v /root/nginx.conf:/etc/nginx/nginx.conf --link flask -d nginx
现在通过服务器的80端口就可以访问到flask应用了
只使用docker的部署完成,下面我们来用docker-compose实现更方便的部署
三、docker-compose部署
docker-compose可以很方便地通过一个文件构建多个镜像、启动多个容器。
安装docker-compose(最新版
sudo curl -L "https://github.com/docker/compose/releases/download/1.23.1/docker-compose-$(uname -s)-$(uname -m)" -o /usr/local/bin/docker-compose chmod +x /usr/local/bin/docker-compose
要使用docker-compose,首先要把每个部分的Dockerfile写好,然后在docker-compose.yml文件中统一构建和启动
整个docker-compose项目的目录结构为
三个目录下分别有它们的Dockerfile和相关的配置文件或代码
flask目录的目录结构和Dockerfile和之前是一样的
mysql的Dockerfile
FROM mysql COPY ./init.sql /docker-entrypoint-initdb.d
init.sql是数据库初始化文件,它会把flask应用需要用到的库和表创建出来。在mysql官方镜像中提供了容器启动时自动执行/docker-entrypoint-initdb.d
文件夹下的脚本的功能(包括shell脚本和sql脚本) 。因此我们只要把初始化文件在镜像启动时复制到/docker-entrypoint-initdb.d
文件夹就可以了
nginx的Dockerfile
FROM nginx COPY ./nginx.conf /etc/nginx/nginx.conf
在之前,我使用挂载的方式将nginx.conf导入nginx容器,这个方法有一个问题就是如果我要把这一套生产环境打包到另一个系统,docker export并不会将挂载的内容一起打包,因此,最好还是在构建镜像的时候就把相关配置加到镜像里。
redis没有需要导入的配置,因此没有它的Dockerfile
docker-compose.yml文件的内容
version: '3' services: mysql: build: ./mysql ports: - "3306:3306" environment: - MYSQL_ROOT_PASSWORD=*** restart: always redis: image: redis:3.2 restart: always flask: build: ./flask ports: - "5000:5000" links: - mysql:mysql - redis:redis-server restart: always nginx: build: ./nginx ports: - "80:80" links: - flask
docker-compose会先从每个服务的build目录下找到Dockerfile,然后依次镜像。所有镜像创建完毕后,再根据yml文件中其他配置依次启动镜像的容器。
当docker-compose启动一次后,只要不更改它自动生成的容器和镜像,再次启动,它就不会再创建镜像,而是启动之前生成的容器。
如果想要更改单个服务的内容,修改Dockerfile的内容、或者删除docker-compose自动生成的容器和镜像,再次启动docker-compose的时候,它会重新构建修改后服务的容器,而其他没有做过修改的服务不变
执行docker-compose up启动所有服务
你的应用已经可以在80端口访问了,enjoy!