写在前面的话

目前解决容器跨主机通信的方案有很多种,这里给出的只是其中的一种,而且还不是最好的方案,不过归根结底,大同小异。在学习 docker swarm 之前,大家可以先看看这种。

啥是 overlay 和 etcd

从这里开始正式引入 overlay 网络这个概念,那啥是 overlay 呢?

可以这样理解,在原本主机通信之上开通的通信隧道,覆盖于原本的主机网络之上,所以叫 overlay。

那 etcd 又是啥?它源于 zookeeper,一个用于分享配置和服务发现的分布式一致性 KV 存储系统。

为啥将这两个东西放在一起?我们的目的就是为了搭建一个能够静态发现,并且能够实现配置共享的容器网络~

etcd 下载地址:

https://github.com/etcd-io/etcd/tags

多主机多容器实践

这里以实例的形式来实现这个功能,需要准备两台安装了 docker 的虚拟机,我这里分别是:

docker-node1:192.168.100.100(之后统称 node1)运行 redis 容器

docker-node2:192.168.100.101(之后统称 node2)运行 flask 容器

具体的拓扑图如下所示,项目就是之前的 flask + redis 项目:

在开始之前,有几点要特别说明:

1. 由于中间会涉及几个端口,建议关闭防火墙,可以省掉很多麻烦。

2. 一定要让两台机器的主机名有意义且不一样,否则莫名其妙的 BUG 调哭你。

3. 为了直接安装 etcd 建议安装 epel 源。

# 修改主机名
hostnamectl set-hostname docker-node1
hostnamectl set-hostname docker-node2

# 关闭防火墙
systemctl stop firewalld.service
systemctl disable firewalld.service

# 安装 epel 源
yum -y install epel-release

截至我写文之前,GITHUB 上面 etcd 最新 TAG 为 3.3.12,而 epel 里面的版本为:3.3.11,所以推荐 epel 源直接 yum 安装。

【1】在 node1 上安装配置 etcd:

yum -y install etcd
cd /etc/etcd/
mv etcd.conf etcd.conf_bak
vim etcd.conf

内容如下:

#[Member]
ETCD_DATA_DIR="/var/lib/etcd/default.etcd"
ETCD_LISTEN_PEER_URLS="http://0.0.0.0:2380"
ETCD_LISTEN_CLIENT_URLS="http://0.0.0.0:2379,http://0.0.0.0:4001"
ETCD_NAME="docker-node1"

#[Clustering]
ETCD_INITIAL_ADVERTISE_PEER_URLS="http://192.168.100.100:2380"
ETCD_ADVERTISE_CLIENT_URLS="http://192.168.100.100:2379,http://192.168.100.100:4001"
ETCD_INITIAL_CLUSTER="docker-node1=http://192.168.100.100:2380,docker-node2=http://192.168.100.101:2380"
ETCD_INITIAL_CLUSTER_TOKEN="etcd-cluster"
ETCD_INITIAL_CLUSTER_STATE="new"

说明:有几个值得注意的地方需要说明:

ETCD_NAME:必须唯一,而且在之后的 ETCD_INITIAL_CLUSTER 参数也会用到。

Clustering 下面的配置除了 ETCD_INITIAL_CLUSTER 里面指定其他节点需要用到其他的 IP,否则都使用本机的 IP。

ETCD_INITIAL_CLUSTER_TOKEN:所有的节点都必须配置一样的值,相当于密钥

【2】在 node2 上面也安装 etcd 和配置 etcd:

yum -y install etcd
cd /etc/etcd/
mv etcd.conf etcd.conf_bak
vim etcd.conf

内容如下:

#[Member]
ETCD_DATA_DIR="/var/lib/etcd/default.etcd"
ETCD_LISTEN_PEER_URLS="http://0.0.0.0:2380"
ETCD_LISTEN_CLIENT_URLS="http://0.0.0.0:2379,http://0.0.0.0:4001"
ETCD_NAME="docker-node2"

#[Clustering]
ETCD_INITIAL_ADVERTISE_PEER_URLS="http://192.168.100.101:2380"
ETCD_ADVERTISE_CLIENT_URLS="http://192.168.100.101:2379,http://192.168.100.101:4001"
ETCD_INITIAL_CLUSTER="docker-node1=http://192.168.100.100:2380,docker-node2=http://192.168.100.101:2380"
ETCD_INITIAL_CLUSTER_TOKEN="etcd-cluster"
ETCD_INITIAL_CLUSTER_STATE="new"

同 node1 的配置,只是需要注意有些标识性的配置改为自己的就行。

简单的做个参数说明:

关键字 说明
ETCD_DATA_DIR 数据存储目录
ETCD_LISTEN_PEER_URLS 与其他节点通信时的监听地址列表,通信协议可以是http、https
ETCD_LISTEN_CLIENT_URLS 与客户端通信时的监听地址列表
ETCD_NAME 节点名称,在也就是后面配置的那个 名字=地址
ETCD_INITIAL_ADVERTISE_PEER_URLS 节点在整个集群中的通信地址列表,可以理解为能与外部通信的ip端口
ETCD_ADVERTISE_CLIENT_URLS 告知集群中其他成员自己名下的客户端的地址列表
ETCD_INITIAL_CLUSTER 集群内所有成员的地址,这就是为什么称之为静态发现,因为所有成员的地址都必须配置
ETCD_INITIAL_CLUSTER_TOKEN 初始化集群口令,用于标识不同集群
ETCD_INITIAL_CLUSTER_STATE 初始化集群状态,new表示新建

【3】在两个节点都启动 etcd 并查看:

systemctl start etcd
systemctl enable etcd

# 查看节点
etcdctl member list

# 监控检查
etcdctl cluster-health

结果如图:

如果发现不通报错,一般都是防火墙问题。

【4】修改两个 docker 的启动配置:

node1:

systemctl stop docker
vim /etc/systemd/system/multi-user.target.wants/docker.service

 ExecStart= 项改成如下内容:如果文件不存在,记得:systemctl enable docker.service

ExecStart=/usr/bin/dockerd -H tcp://0.0.0.0:2375 -H unix:///var/run/docker.sock --cluster-store=etcd://192.168.100.100:2379 --cluster-advertise=192.168.100.100:2375

启动 docker:

systemctl daemon-reload
systemctl start docker

node2:

systemctl stop docker
vim /etc/systemd/system/multi-user.target.wants/docker.service

 ExecStart= 项改成如下内容:

ExecStart=/usr/bin/dockerd -H tcp://0.0.0.0:2375 -H unix:///var/run/docker.sock --cluster-store=etcd://192.168.100.101:2379 --cluster-advertise=192.168.100.101:2375

启动 docker:

systemctl daemon-reload
systemctl start docker

【5】 在 node1 上面我们创建一个 overlay 网络:

# 创建网络
docker network create -d overlay demo

# 查看
docker network ls

结果如图:

此时在 node2 上面查看网络应该也是存在的~

如果你发现 node2 没有同步或者在 node1 上面直接创建就报 docker swarm 错误,建议重启两个机器的 etcd 服务和 docker 服务,一般就能解决。

【6】我们此时可以创建两个测试容器测试网络的连通性:

node1:

docker run -d --name b1 --net demo busybox sh -c "while true; do sleep 3600; done"

node2:

docker run -d --name b2 --net demo busybox sh -c "while true; do sleep 3600; done"

node1 上测试连通性:

docker container exec b1 ping b2

可以发现已经可以正常通信了。

【7】导出和导入镜像:

这里专门提供了一种导出和导入镜像的方法,因为在 node2 上面没有 Flask 项目的镜像,可以在 node1 上的镜像导出,再到 node2 上面导入,可以节省我们下载的时间。

node1:

docker save python:2.7 dylan/flask-demo:v1.0 -o image.tar

将镜像传到 node2 上面,然后在 node2 执行:

docker load -i image.tar

完成以后在 node2 上面就有了这些镜像。

【8】运行项目:

node1:运行 redis 容器

docker run -d --name redis --network demo redis

node2:运行 flask 容器

docker run -d --name flask-demo --network demo -it -e REDIS_HOST="redis" -p 8080:5000 dylan/flask-demo:v1.0

【9】访问测试:

值得我们注意得是,在这里也没有指定 --link 参数。

小结

总的来说,这种方法还是很复杂的,而且由于是静态发现,每一次增加减少节点都会去修改配置重启 etcd,非常不方便。

【06】循序渐进学 docker:跨主机通信的更多相关文章

  1. Docker跨主机通信(九)--技术流ken

    容器网络 在前面的博客中已经详细讲解了几种网络方案: none, host, bridge,user-defined.但是他们只是解决了单个主机间的容器的通信问题,并不能实现多个主机容器之间的通信.本 ...

  2. Docker跨主机通信(九)

    容器网络 在前面的博客中已经详细讲解了几种网络方案: none, host, bridge,user-defined.但是他们只是解决了单个主机间的容器的通信问题,并不能实现多个主机容器之间的通信.本 ...

  3. docker跨主机通信扁平化网络的设计与实现

    端口映射.ovs. fannel,weave 1.使用网桥实现跨主机容器连接 使用Open vSwitch实现跨主机容器连接

  4. Docker跨主机通信之路由

    一.实验环境: 主机名 主机IP Docker0_IP Docker1 192.168.88.130 172.17.0.1 Docker2 192.168.88.131 172.18.0.1 二.实验 ...

  5. docker跨主机通信-overlay

    使用consul 1,让两个网络环境下的容器互通,那么必然涉及到网络信息的同步,所以需要先配置一下consul. 直接运行下面命令.启动consul. docker run -d -p 8500:85 ...

  6. Docker跨主机网络——overlay

    前言 在Docker网络--单host网络一文中,我为大家总结了Docker的单机网络相关知识和操作,单机网络比较容易.本文我为大家总结Docker跨主机通信相关知识.同样本文大部分内容以CloudM ...

  7. Docker跨主机网络实践

    Docker使用中网络管理是最麻烦的,在项目初始化前期就需要进行合理的规划,如果在比较理想的单主机的网络通信是比较简单的,但如果涉及到跨主机的网络就需要使用docker自带的overlay netwo ...

  8. overlay 如何实现跨主机通信?- 每天5分钟玩转 Docker 容器技术(52)

    上一节我们在 host1 中运行了容器 bbox1,今天将详细讨论 overlay 网络跨主机通信的原理. 在 host2 中运行容器 bbox2: bbox2 IP 为 10.0.0.3,可以直接 ...

  9. Docker 网络管理及容器跨主机通信

    1.网络模式 docker支持四种网络模式,使用--net选项指定: host,--net=host,如果指定此模式,容器将不会获得一个独立的network namespace,而是和宿主机共用一个. ...

随机推荐

  1. VS Code 在新Tabs打开文件

    添加如下设置即可 "workbench.editor.enablePreview": false

  2. axis2 webService开发指南(2)

    1  Axis2的简单WebService示例 1.1 新建一个web工程,创建一个类Greeting,用于当作webservice服务 代码如下: package amyservices; impo ...

  3. Spring IOC的配置使用

    1.1.1 XML配置的结构一般配置文件结构如下: <beans> <import resource=”resource1.xml” /> <bean id=”bean1 ...

  4. nodejs开发工具

    我选择的是Hbuilder作为node项目的开发工具.   先在Hbuilder 里面安装nodeEclipse插件,然后重启工具. 点击添加项目,选择其他选项,出现下图选项,然后选择圈住的选项点击下 ...

  5. Superset安装

    Superset version 1.8.5     # Install superset   pip install cairocffi   pip install superset     yum ...

  6. www请求用到的Unescape

    //发送请求          WWW w = new WWW(url); //web服务器返回          yield return w;        if (!string.IsNullO ...

  7. iOS开发时间控件怎么强制24小时制(小技巧)

    1)当你的format格式是 NSDateFormatter* dateFormatter = [[[NSDateFormatter alloc] init] autorelease];[dateFo ...

  8. Linux下软件常见安装方式

    pasting  分类: Linux2007-12-08 16:31 1909人阅读 评论(0) 收藏 举报 linuxredhat脚本文档managerfile        Linux下软件安装主 ...

  9. YII2 全局设定 默认参数

    $gridviewSettings = [ 'export' => false, 'responsive' => true, 'floatHeader' => true, 'floa ...

  10. jQuary总结8:动画操作-自定义动画

    1 自定义动画 语法: jQuery对象.animate(json,[speed],[easing],[callback]) 参数详解: -1 json:要执行动画的CSS属性,带数字(必选) -2  ...