Docker应用部署

传统在linux中安装软件非常麻烦,还需要配置,有了docker我们可以快速安装软件搭建环境。

  • 容器内地网络服务和外部机器不能直接通信
  • 外部机器和宿主可以直接通信
  • 宿主机和容器可以直接通信
  • 当容器中的网络服务需要被外部机器访问时,可以将容器中的提供服务的端口映射到宿主机的端口中。外部机器就可以访问该宿主机的该端口,从而间接访问容器的服务

这种操作称作端口映射

image-20210729112507701

Mysql部署

Docker中部署Mysql,外部客户端访问容器的Mysql

  1. 拉取mysql镜像
docker pull mysql:5.6
  1. 在root创建docker目录,docker目录中创建mysql目录
cd ~

mkdir docker

cd docker

mkdir mysql
  1. 创建mysql容器
# $PWD代表当前目录
# -e 设置环境变量
docker run -id \
-p 3307:3306 \
--name=c_mysql \
-v $PWD/conf:/etc/mysql/conf.d \
-v $PWD/logs:/logs \
-v $PWD/data:/var/lib/mysql \
-e MYSQL_ROOT_PASSWORD=123456 \ 
mysql:5.7
  1. 进入mysql容器,操作mysql
docker exec -it c_mysql /bin/bash

root@1946996a10dd:/# mysql -uroot -p123456

mysql> show databases;
+--------------------+
| Database           |
+--------------------+
| information_schema |
| mysql              |
| performance_schema |
+--------------------+
3 rows in set (0.00 sec)
  1. 客户端连接docker容器mysql

    • 访问映射的端口,宿主的3307

      image-20210729115137017

Tomcat部署

Docker中部署Tomcat的容器,通过外部访问Docker中的Tomcat部署的项目

  1. 拉取tomcat
docker pull tomcat

2.创建容器,设置端口映射和目录映射

cd ~/docker
mkdir tomcat
cd tomcat

#创建容器映射宿主机8080到docker容器的8080
#自动在docker容器中创建了/usr/local/tomcat/webapps
#挂载到了~/docker/tomcat中
docker run -id --name=c_tomcat \
-p 8080:8080 \
-v $PWD:/usr/local/tomcat/webapps \
tomcat
  1. 宿主机的数据卷中创建项目目录,写一个网页

  1. 外部机器访问tomcat部署的项目网页

image-20210729125757904

Nginx部署

默认的配置运行Nginx

  1. 首先运行docker run -it --name=nginx -p:80:80 nginx

创建了1个nginx的容器并运行

  1. docker exec -it nginx /bin/bash

    进入容器可查看到几个重要的文件

    • 配置文件:nginx.conf 在 /etc/nginx/nginx.conf
    • 日志文件: /var/log/nginx/access.log /var/log/nginx/error.log
  2. cat nginx.conf查看内容

     1 root@dc048fc59765:/var/log/nginx# cat /etc/nginx/nginx.conf 
     2 
     3 user  nginx;
     4 worker_processes  1;
     5 
     6 error_log  /var/log/nginx/error.log warn;
     7 pid        /var/run/nginx.pid;
     8 
     9 
    10 events {
    11     worker_connections  1024;
    12 }
    13 
    14 
    15 http {
    16     include       /etc/nginx/mime.types;
    17     default_type  application/octet-stream;
    18 
    19     log_format  main  '$remote_addr - $remote_user [$time_local] "$request" '
    20                       '$status $body_bytes_sent "$http_referer" '
    21                       '"$http_user_agent" "$http_x_forwarded_for"';
    22 
    23     access_log  /var/log/nginx/access.log  main;
    24 
    25     sendfile        on;
    26     #tcp_nopush     on;
    27 
    28     keepalive_timeout  65;
    29 
    30     #gzip  on;
    31 
    32     include /etc/nginx/conf.d/*.conf;
    33 }

    发现第32行,配置了一个子配置文件,进入conf.d发现有一个default.conf文件

  3. cat打开default.conf文件:

    从default.conf中11行可以看出,index页面在/usr/share/nginx/html

    root@dc048fc59765:/etc/nginx/conf.d# cat default.conf
    server {
        listen       80;
        listen  [::]:80;
        server_name  localhost;
    
        #charset koi8-r;
        #access_log  /var/log/nginx/host.access.log  main;
    
        location / {
            root   /usr/share/nginx/html;
            index  index.html index.htm;
        }
    
        #error_page  404              /404.html;
    
        # 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;
        }
    
        # proxy the PHP scripts to Apache listening on 127.0.0.1:80
        #
        #location ~ \.php$ {
        #    proxy_pass   http://127.0.0.1;
        #}
    
        # pass the PHP scripts to FastCGI server listening on 127.0.0.1:9000
        #
        #location ~ \.php$ {
        #    root           html;
        #    fastcgi_pass   127.0.0.1:9000;
        #    fastcgi_index  index.php;
        #    fastcgi_param  SCRIPT_FILENAME  /scripts$fastcgi_script_name;
        #    include        fastcgi_params;
        #}
    
        # deny access to .htaccess files, if Apache's document root
        # concurs with nginx's one
        #
        #location ~ /\.ht {
        #    deny  all;
        #}
    }
    
    root@dc048fc59765:/etc/nginx/conf.d#
  4. 外部访问宿主机的地址ip,看到nginx默认的欢迎页

    img

数据卷方式运行

用挂载目录到数据卷的方式自定义配置文件,启动页等运行nginx

  1. 宿主机先创建nginx数据目录
[root@VM-0-9-centos test-project]# cd ~/docker/
[root@VM-0-9-centos docker]# mkdir nginx
[root@VM-0-9-centos docker]# cd nginx/
[root@VM-0-9-centos nginx]# mkdir conf
[root@VM-0-9-centos nginx]# cd conf
  1. 创建nginx.conf文件
vi nginx.conf
user  nginx;
worker_processes  1;

error_log  /var/log/nginx/error.log warn;
pid        /var/run/nginx.pid;


events {
    worker_connections  1024;
}


http {
    include       /etc/nginx/mime.types;
    default_type  application/octet-stream;

    log_format  main  '$remote_addr - $remote_user [$time_local] "$request" '
                      '$status $body_bytes_sent "$http_referer" '
                      '"$http_user_agent" "$http_x_forwarded_for"';

    access_log  /var/log/nginx/access.log  main;

    sendfile        on;
    #tcp_nopush     on;

    keepalive_timeout  65;

    #gzip  on;

    include /etc/nginx/conf.d/*.conf;
}
  1. 启动nginx的容器,配置映射
docker run -it --name=nginx \
-p 80:80 \
-v $PWD/conf/nginx.conf:/etc/nginx/nginx.conf \
-v $PWD/logs:/var/log/nginx \
-v $PWD/html:/usr/share/nginx/html \
-v $PWD/conf.d::/etc/nginx/conf.d \
nginx

#
docker run --name nginx \
-id -p 80:80 \ 
-v $PWD/html:/usr/share/nginx/html \
-v $PWD/conf/nginx.conf:/etc/nginx/nginx.conf \
-v $PWD/logs:/var/log/nginx \
-v $PWD/conf.d:/etc/nginx/conf.d \
nginx

#
docker run --name nginx -id -p 80:80 -v $PWD/html:/usr/share/nginx/html -v $PWD/conf/nginx.conf:/etc/nginx/nginx.conf -v $PWD/logs:/var/log/nginx -v $PWD/conf.d:/etc/nginx/conf.d nginx


#进入docker nginx bash
docker exec -it nginx /bin/bash

参数说明:

  • -p 80:80:将容器的 80端口映射到宿主机的 80 端口。
  • -v $PWD/conf/nginx.conf:/etc/nginx/nginx.conf:将主机当前目录下的 /conf/nginx.conf 挂载到容器的 :/etc/nginx/nginx.conf。配置目录
  • -v $PWD/logs:/var/log/nginx:将主机当前目录下的 logs 目录挂载到容器的/var/log/nginx。日志目录
  1. 外部访问Nginx

image-20210729135341502

部署Redis

  1. 搜索redis镜像
docker search redis
  1. 拉取redis镜像
docker pull redis:5.0
  1. 创建容器,设置端口映射
docker run -id --name=c_redis -p 6379:6379 redis:5.0
  1. 使用外部机器连接redis
./redis-cli.exe -h 192.168.149.135 -p 6379

Docker容器访问宿主的服务

使用nginx作反向代理,其中nginx是使用docker方式运行:

$ docker run -d --name nginx $PWD:``/etc/nginx` `-p 80:80 -p 443:443 nginx:1.15

需要代理的API服务运行在宿主机的 1234 端口, nginx.conf 相关配置如下:

server {
 ...
 
 location /api {
  proxy_pass http://localhost:1234
 }
 ...
}

结果访问的时候发现老是报 502 Bad Gateway 错误,错误日志显示无法连接到upstream。

仔细想一想, nginx.conf 中的 localhost 似乎有问题。由于nginx是运行在docker容器中的,这个 localhost 是容器的localhost,而不是宿主机的localhost。

到这里,就出现了本文要解决的问题:如何从容器中访问到宿主机的网络?通过搜索网络,有如下几种方法:

使用宿主机IP

在安装Docker的时候,会在宿主机安装一个虚拟网关 docker0 ,我们可以使用宿主机在 docker0 上的IP地址来代替 localhost 。

首先,使用如下命令查询宿主机IP地址:

$ ip addr show docker0
3: docker0: <BROADCAST,MULTICAST,UP,LOWER_UP> mtu 1500 qdisc noqueue state UP group default
  link/ether 02:42:d5:4c:f2:1e brd ff:ff:ff:ff:ff:ff
  inet 172.17.0.1/16 scope global docker0
    valid_lft forever preferred_lft forever
  inet6 fe80::42:d5ff:fe4c:f21e/64 scope link
    valid_lft forever preferred_lft forever

可以发现宿主机的IP是 172.17.0.1 ,那么将 proxy_pass http://localhost:1234 改为 proxy_pass http://172.17.0.1:1234 就可以解决 502 Bad Gateway 错误。

但是,不同系统下宿主机的IP是不同的,例如Linux下一般是 172.17.0.1 , macOS下一般是 192.168.65.1 ,并且这个IP还可以更改。所以使用IP配置 nginx.conf ,不能跨环境通用。

使用host网络

Docker容器运行的时候有 host 、 bridge 、 none 三种网络可供配置。默认是 bridge ,即桥接网络,以桥接模式连接到宿主机; host 是宿主网络,即与宿主机共用网络; none 则表示无网络,容器将无法联网。

当容器使用 host 网络时,容器与宿主共用网络,这样就能在容器中访问宿主机网络,那么容器的 localhost 就是宿主机的 localhost 。

在docker中使用 --network host 来为容器配置 host 网络:

$ docker run -d --name nginx --network host nginx

上面的命令中,没有必要像前面一样使用 -p 80:80 -p 443:443 来映射端口,是因为本身与宿主机共用了网络,容器中暴露端口等同于宿主机暴露端口。

使用host网络不需要修改 nginx.conf ,仍然可以使用 localhost ,因而通用性比上一种方法好。但是,由于 host 网络没有 bridge 网络的隔离性好,使用 host 网络安全性不如 bridge 高。

总结

本文提出了使用宿主机IP和使用host网络两种方法,来实现从容器中访问宿主机的网络。两种方法各有优劣,使用宿主机IP隔离性更好,但通用性不好;使用host网络,通用性好,但带来了暴露宿主网络的风险。