本文主要内容关于使用docker-compose实践部署后端django-rest-framework和前端vue.js应用。记录其中遇到的一些坑以及解决办法。html
系统:Ubuntu 16.04(阿里云)
代码中用户名:test前端
# install docker ## prepare echo 'Preparing...' sudo apt update sudo apt upgrade -y sudo apt install -y linux-image-extra-$(uname -r) linux-image-extra-virtual ## docker echo 'Installing docker...' sudo apt remove -y docker-ce docker-engine docker.io wget -qO- http://acs-public-mirror.oss-cn-hangzhou.aliyuncs.com/docker-engine/internet | sh sudo apt autoremove -y sudo usermod -aG docker ${USER} ## docker Aliyun accelerator ## https://cr.console.aliyun.com/#/accelerator echo 'Configuring docker registry mirrors...' sudo mkdir -p /etc/docker sudo tee /etc/docker/daemon.json <<-'EOF' { "registry-mirrors": ["https://<your-own>.mirror.aliyuncs.com"] } EOF exists(){ command -v "$1" >/dev/null 2>&1 } echo 'Installing docker-compose...' if ! exists pip; then sudo apt install python-pip fi sudo python `which pip` install docker-compose echo 'Restarting docker...' sudo systemctl daemon-reload sudo systemctl restart docker
. ├── .env // 环境变量 ├── docker-compose.yml ├── backend // 放置后台django文件 ├── frontend // 放置前端vue编译后代码 └── nginx // nginx相关配置 ├── backend.conf ├── Dockerfile └── frontend.conf
docker-compose.yml
version: '3' services: web: restart: always build: ./backend expose: - "8000" volumes: - ./backend:/code env_file: .env links: - db depends_on: - db command: ["/code/wait-for-it.sh", "db:3306", "--", "bash","startup.sh"] nginx: restart: always build: ./nginx ports: - "80:80" volumes: - ./frontend:/usr/share/nginx/html/frontend:ro - ./backend/public:/usr/share/nginx//html/backend/public:ro links: - web depends_on: - web db: restart: always image: mysql:latest env_file: .env volumes: - ./data/initsql:/docker-entrypoint-initdb.d - ./data/db:/var/lib/mysql command: [mysqld, --character-set-server=utf8, --collation-server=utf8_unicode_ci]
第一次使用docker-compose部署,从网上参考了许多例子。可是因为docker-compose个版本的语法改动不小,遇到不少坑:vue
有些例子使用
volumes_from
,可是version 3已经删除该设置项更新变化。
官方推荐在根节点(与services
同级)下配置volumes
,可是无法映射到host主机的文件,有个插件local-persist
能够作到,可是不想依赖第三方插件。只能使用麻烦点的写法:在每一个service的volume
设置项里重复映射host主机的文件pythonweb: volumes: - ./backend:/code nginx: volumes: - ./backend/public:/usr/share/nginx//html/backend/public:ro
使用links实现容器间通讯。配置须要访问的容器于links配置项下,没什么毛病。
实现容器按顺序启动,须要用到
depends_on
。mysql可是会有一个问题:
depends_on
只保证启动顺序,而咱们的实际需求是:web容器启动的commands必须等到mysql彻底启动,web容器的command才能够执行。由于咱们加了restart: always
,因此会致使web容器不断重启直到db容器启动完成。linux个人解决办法是:给web容器的启动命令加上对mysql服务可用的查询,使用了一个查询脚本wait-for-it:nginx
command: ["/code/wait-for-it.sh", "db:3306", "--", "bash","startup.sh"]
会有多个容器使用相同环境变量的状况,因此都放在
.env
文件里gitDEBUG=false MYSQL_HOST=db MYSQL_DATABASE=mydb MYSQL_ROOT_PASSWORD=mypass
能够定义volume持久化数据库文件:github
volumes:- 若是有初始数据须要导入,能够定义volume映射到`/docker-entrypoint-initdb.d`:volumes:web
- ./data/initsql:/docker-entrypoint-initdb.d注意: 该配置只有在`/var/lib/mysql/`下的`mysql`目录不存在时才会生效。也就是说,一旦容器启动过一次,以后就在也不会导入`/docker-entrypoint-initdb.d`里的文件,除非手动清空`/var/lib/mysql/`(或host主机的`./data/db`目录)。
Dockerfile
FROM nginx:alpine RUN rm /etc/nginx/conf.d/default.conf ADD frontend.conf /etc/nginx/conf.d/ ADD backend.conf /etc/nginx/conf.d/
frontend.conf
- vue app build filesserver { listen 80 deferred; server_name new.bylie.cn; root /usr/share/nginx/html/frontend; location / { try_files $uri $uri/ /index.html =404; } # redirect server error pages to the static page /50x.html # error_page 500 502 503 504 /50x.html; location = /50x.html { root /usr/share/nginx/html; } }
backend.conf
- django appserver { # use 'listen 80 deferred;' for Linux # use 'listen 80 accept_filter=httpready;' for FreeBSD listen 80 deferred; client_max_body_size 5M; # set the correct host(s) for your site server_name service.bylie.cn; keepalive_timeout 5; location /public { root /usr/share/nginx/html/backend; } location / { proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; # enable this if and only if you use HTTPS # proxy_set_header X-Forwarded-Proto https; proxy_set_header Host $http_host; # we don't want nginx trying to do something clever with # redirects, we set the Host: header above already. proxy_redirect off; proxy_pass http://web:8000; } error_page 500 502 503 504 /50x.html; location = /50x.html { root html; } }备份数据库到host主机
docker-compose exec db sh -c 'exec mysqldump -uroot -p"$MYSQL_ROOT_PASSWORD" --databases ${MYSQL_DATABASE}' > ./backup/database.sql
比较简单
Dockerfile
FROM python:3 ENV PYTHONUNBUFFERED 1 RUN mkdir /code WORKDIR /code ADD requirements.txt /tmp/ RUN pip install -r /tmp/requirements.txt启动脚本
startup.sh
#!/usr/bin/env bash python manage.py collectstatic --noinput && python manage.py migrate && gunicorn django_web_app.wsgi:application -w 2 -b :8000当容器运行起来以后,可能须要导入一些初始数据或者fixtures,我写了一个init的django custom command,手动执行该command:
docker-compose exec web python manage.py init