Docker-Compose环境搭建和部署SpringBoot项目

前言

fighting


Docker使用nsenter工具进入容器

1.下载nsenter工具

wget https://mirrors.edge.kernel.org/pub/linux/utils/util-linux/v2.33/util-linux-2.33.tar.gz

2.解压util-linux-2.33.tar.gz

tar -zxvf util-linux-2.33.tar.gz

3.进入cd util-linux-2.33目录,输入./configure --without-ncurses命令进行check

4.输入make nsenter,对nsenter进行编译

5.复制nsenter到/usr/local/bin

cp nsenter /usr/local/bin

6.查看nsenter的版本

[root@localhost util-linux-2.33]# nsenter --version
nsenter,来自 util-linux 2.33

7.docker inspect命令会提取出容器或者镜像最顶层的元数据,我们可以通过PID=$(docker inspect --format "{{ .State.Pid}}" <container id>)获取容器的进程id,然后再通过nsenter --target $PID --mount --uts --ipc --net --pid进入到容器中。

8.获取12dfffb03859容器的pid,docker inspect --format "{{.State.Pid}}" 12dfffb03859

[root@localhost util-linux-2.33]# docker inspect --format "{{.State.Pid}}" 12dfffb03859
1894

9.nsenter --target 1894 --mount --uts --ipc --net --pid进入容器.

10.我们把以上繁琐的操作写成docker-enter.sh

#!/bin/sh

if [ -e $(dirname "$0")/nsenter ]; then
  # with boot2docker, nsenter is not in the PATH but it is in the same folder
  NSENTER=$(dirname "$0")/nsenter
else
  NSENTER=nsenter
fi

if [ -z "$1" ]; then
  echo "Usage: `basename "$0"` CONTAINER [COMMAND [ARG]...]"
  echo ""
  echo "Enters the Docker CONTAINER and executes the specified COMMAND."
  echo "If COMMAND is not specified, runs an interactive shell in CONTAINER."
else
  PID=$(docker inspect --format "{{.State.Pid}}" "$1")
  if [ -z "$PID" ]; then
    exit 1
  fi
  shift

  OPTS="--target $PID --mount --uts --ipc --net --pid --"

  if [ -z "$1" ]; then
    # No command given.
    # Use su to clear all host environment variables except for TERM,
    # initialize the environment variables HOME, SHELL, USER, LOGNAME, PATH,
    # and start a login shell.
#"$NSENTER" $OPTS su - root
"$NSENTER" $OPTS /bin/su - root
  else
    # Use env to clear all host environment variables.
    "$NSENTER" $OPTS env --ignore-environment -- "$@"
  fi
fi

11.在/etc/profile设置别名alias docker-enter=/usr/local/docker-sh/docker-enter.sh,然后source /etc/profile保存配置,再alias docker-enter查看设置别名是否生效

image.png

  1. docker-enter 9393ed00b852,进入容器成功。
    image.png

13.感觉用nsenter有点麻烦,其实可以用docker exec -it 12dfffb03859 /bin/bash,如果出现stat /bin/bash: no such file or directory的错误,这是由于容器中的PATH 路径问题,使用/bin/su 即可。

[root@localhost util-linux-2.33]# docker exec -it 12dfffb03859 /bin/su
/ # ls
bin       etc       lib64     proc      sbin      tmp
demo.jar  home      media     root      srv       usr
dev       lib       mnt       run       sys       var
/ # 

Docker Compose介绍

Dockerfile可以让用户管理一个单独的应用容器;而Docker Compose则允许用户在一个模板(yaml格式)中定义一组相关联的应用容器(被称为一个project,即项目),例如一个web服务容器再加上redis服务容器,nginx服务容器等。

docker compose.png

搭建Docker-Compose环境

1.下载安装docker-compose

#下载
sudo curl -L https://github.com/docker/compose/releases/download/1.20.0/docker-compose-`uname -s`-`uname -m` -o /usr/local/bin/docker-compose
#安装
chmod +x /usr/local/bin/docker-compose
#查看版本
docker-compose version
image.png

2.下载docker补全命令

#安装
yum install bash-completion
#下载docker-compose脚本
curl -L https://raw.githubusercontent.com/docker/compose/$(docker-compose version --short)/contrib/completion/bash/docker-compose > /etc/bash_completion.d/docker-compose
image.png

Docker-compose部署Python Flask项目

1.创建app.py

import time
import redis
from flask import Flask

app = Flask(__name__)
cache = redis.Redis(host='redis', port=6379)

def get_hit_count():
        retries = 5
        while True:
                try:
                        return cache.incr('hits')
                except redis.exceptions.ConnectionError as exc:
                        if retries == 0:
                                raise exc
                        retries -= 1
                        time.sleep(0.5)


@app.route('/')
def hello():
        count = get_hit_count()
        return 'hello python! i have been seen {} times\n'.format(count)

if __name__ == "__main__":
        app.run(host="0.0.0.0", debug=True)

2.同目录下创建requirements.txt文件,添加项目依赖的python包

flask
redis

3.创建网络cmazxiaoma_net

[root@localhost python-flask]# docker network create cmazxiaoma_net
392423d2779c8ef18d1c4bbc1d1a9142ae1b2fc63bc73e716bdd9e76013387cf
[root@localhost python-flask]# docker network ls

4.创建Dockerfile

FROM python:3.4-alpine
maintainer cmazxiaoma
ADD . /python-code #将当前目录的所有文件拷贝至/python-code
WORKDIR /python-code #设置工作目录为/python-code
RUN pip install -r requirements.txt #安装python依赖包
CMD ["python", "app.py"] #容器启动时允许app.py

5.创建docker-compose.yml

version: '2'
services:
        web:
                build: . # 构建容器
                ports:
                        - "9021:5000"
                volumes:
                        - .:/python-code  #将当前目录挂载到web容器中的/python-code
                depends_on: # redis服务先启动
                        - redis
                networks:
                        - custom_net
        redis:
                image:  "redis:alpine"
                container_name: redis_container
                restart:         always
                ports:
                        - 6380:6379
                expose:
                        - 6379
                networks:
                        - custom_net
networks:
        custom_net:
                external:
                        name: cmazxiaoma_net #设置网络
  1. 把项目目录添加到selinux白名单
chcon -Rt svirt_sandbox_file_t python-flask

7.docker-compose up -d后台启动应用

image.png

image.png

8.docker-compose run web ping redis测试同一应用下的redis和web服务是否能ping通。

image.png

9.docker-compose相关的命令如下:

#查看帮助
docker-compose -h

# -f  指定使用的 Compose 模板文件
# 默认为 docker-compose.yml,可以多次指定。
docker-compose -f docker-compose.yml up -d 

#启动所有容器,-d 将会在后台启动并运行所有的容器
docker-compose up -d

#停用移除所有容器以及网络相关
docker-compose down

#查看服务容器的输出
docker-compose logs

#列出项目中目前的所有容器
docker-compose ps

#构建(重新构建)项目中的服务容器。服务容器一旦构建后,将会带上一个标记名.
#例如对于 web 项目中的一个 db 容器,可能是 web_db。
#可以随时在项目目录下运行 docker-compose build 来重新构建服务
docker-compose build

# 不带缓存的构建。
docker-compose build --no-cache

#拉取服务依赖的镜像
docker-compose pull

#重启项目中的服务
docker-compose restart

#删除所有(停止状态的)服务容器。
#推荐先执行 docker-compose stop 命令来停止容器。
docker-compose rm 

#在指定服务上执行一个命令。
docker-compose run ubuntu ping docker.com

#设置指定服务运行的容器个数。通过 service=num 的参数来设置数量
docker-compose scale web=3 db=2

#启动已经存在的服务容器。
docker-compose start

#停止已经处于运行状态的容器,但不删除它。
#通过 docker-compose start 可以再次启动这些容器。
docker-compose stop

Docker-compose部署SpringBoot项目

1.项目目录


目录结构.png

2.nginx.conf

server {
    listen 80;
    charset utf-8;
    access_log off;

    location / {
        proxy_pass http://app:8080;
        proxy_set_header Host $host:$server_port;
        proxy_set_header X-Forwarded-Host $server_name;
        proxy_set_header X-Real-IP $remote_addr;
        proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
    }

    location /static {
        access_log   off;
        expires      30d;

        alias /app/static;
    }
}
  1. docker-compose.yml
version: '3'
services:
  nginx:
   container_name: v-nginx
   image: nginx:1.13
   restart: always
   ports:
   - 81:80
   - 444:443
   volumes:
   - ./nginx/conf.d:/etc/nginx/conf.d

  mysql:
   container_name: v-mysql
   image: mysql/mysql-server:5.7
   environment:
    MYSQL_DATABASE: root
    MYSQL_ROOT_PASSWORD: root
    MYSQL_ROOT_HOST: '%'
   ports:
   - "3308:3306"
   restart: always

  app:
    restart: always
    build: .
    working_dir: /jenkins-demo
    volumes:
      - .:/jenkins-demo
      - ~/.m2:/root/.m2
    expose:
      - "8080"
    depends_on:
      - nginx
      - mysql
    command: mvn clean spring-boot:run -Dspring-boot.run.profiles=docker

4.Dockerfile

FROM maven:3.5-jdk-8
MAINTAINER cmazxiaoma
WORKDIR /jenkins-demo
RUN mkdir /root/.m2
COPY settings.xml /root/.m2/
RUN mkdir -p /usr/local/mvn-resource/repository && \
cd /usr/local/mvn-resource/repository && \
echo $(mvn --version) && \
mvn help:system && \
chmod 777 /usr/local/mvn-resource && \
chmod 777 /jenkins-demo && \

5.setting.xml中需要更改的配置


repository.png
alimaven.png

6.application-docker.properties

server.port=8080
spring.datasource.driver-class-name=com.mysql.jdbc.Driver
spring.datasource.url=jdbc:mysql://mysql:3306/docker-compose?serverTimezone=Asia/Shanghai&useUnicode=true&characterEncoding=utf-8&useSSL=false
spring.datasource.username=root
spring.datasource.password=root
spring.jpa.hibernate.ddl-auto=none
spring.jpa.show-sql=true
spring.jpa.database-platform=org.hibernate.dialect.MySQL5InnoDBDialect

spring.resources.add-mappings=true

7.添加linux规则,把要挂载的目录~/.m2添加到selinux白名单。

chcon -Rt svirt_sandbox_file_t ~/.m2
chcon -Rt svirt_sandbox_file_t /jenkins-demo

8.启动应用docker-compose up,浏览界面成功!

image.png

nginx上面的图片.png

Docker-Compose顺序问题

docker-compose 虽然可以通过 depends_on来定义服务启动的顺序,但是无法确定服务是否启动完成。因此会出现这样一个现象,redis服务启动比较慢,当项目已经启动起来,但是redis还没有初始化好,这样当项目连接redis的时候就会出现连接数据库的异常。

针对这样的问题,有两种解决方案:

  • 足够的容错和重试机制,比如连接redis,在初次连接不上的时候,服务消费者可以不断重试,直到连接上服务。也就是在服务中定义:restart: always

  • 同步等待,使用wait-for-it.sh或者其他shell脚本将当前服务启动阻塞,直到被依赖的服务加载完毕。


参考文章

1.Spring Boot 2.0(五):Docker Compose + Spring Boot + Nginx + Mysql 实践
2.Docker -v 对挂载的目录没有权限 Permission denied解决办法


尾言

最近很焦虑,发现自己很多东西都不会。不要扰乱他人的心志,不要动摇自己的决心。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,313评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,369评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,916评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,333评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,425评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,481评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,491评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,268评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,719评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,004评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,179评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,832评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,510评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,153评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,402评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,045评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,071评论 2 352

推荐阅读更多精彩内容