前半部分主要为 Consul 的一些介绍,若已经了解 Consul,可以直接跳转到:
Consul
是 HashiCorp 公司推出的开源工具,用于实现分布式系统的服务发现与配置。Consul 是分布式的、高可用的、 可横向扩展的。它具备以下特性:
Key/Value
存储: 应用程序可以根据自己的需要使用 Consul 提供的 Key/Value 存储。 Consul 提供了简单易用的 HTTP 接口,结合其他工具可以实现动态配置、功能标记、领袖选举等等功能。Consul 集群间使用了 Gossip
协议通信和 raft 一致性算法。上面这张图涉及到了很多术语:
这里先简单介绍一下 Gossip
协议的执行过程:
Gossip 过程是由种子节点发起,当一个种子节点有状态需要更新到网络中的其他节点时,它会随机的选择周围几个节点散播消息,收到消息的节点也会重复该过程,直至最终网络中所有的节点都收到了消息。这个过程可能需要一定的时间,由于不能保证某个时刻所有节点都收到消息,但是理论上最终所有节点都会收到消息,因此它是一个最终一致性协议。
现在,我们通过一个具体的实例来深入体会一下 Gossip 传播的完整过程: 为了表述清楚,我们先做一些前提设定 1、Gossip 是周期性的散播消息,把周期限定为 1 秒 2、被感染节点随机选择 k 个邻接节点(fan-out)散播消息,这里把 fan-out 设置为 3,每次最多往 3 个节点散播。 3、每次散播消息都选择尚未发送过的节点进行散播 4、收到消息的节点不再往发送节点散播,比如 A -> B,那么 B 进行散播的时候,不再发给 A。 这里一共有 16 个节点,节点 1 为初始被感染节点,通过 Gossip 过程,最终所有节点都被感染:
注意要点:
名称 | 优点 | 缺点 | 接口 | 一致性算法 |
---|---|---|---|---|
zookeeper | 1.功能强大,不仅仅只是服务发现 2.提供 watcher 机制能实时获取服务提供者的状态 3.dubbo 等框架支持 |
1.没有健康检查 2.需在服务中集成 sdk,复杂度高 3.不支持多数据中心 |
sdk | Paxos |
consul | 1.简单易用,不需要集成 sdk 2.自带健康检查 3.支持多数据中心 4.提供 web 管理界面 |
1.不能实时获取服务信息的变化通知 | http/dns | Raft |
etcd | 1.简单易用,不需要集成 sdk 2.可配置性强 |
1.没有健康检查 2.需配合第三方工具一起完成服务发现 3.不支持多数据中心 |
http | Raft |
首先 clone
本项目到本地:
$ git clone git@github.com:elfgzp/python-consul-demo.git
Consul 单机部署所需要用到的文件如下:
docker-compose
配置文件docker-compose
配置文件可以看到两个 docker-compose 文件均暴露了 8 个端口。根据官方文档,这些端口的作用如下:
Ports Used
Consul requires up to 6 different ports to work properly, some on TCP, UDP, or both protocols. Below we document the requirements for each port.
- Server RPC (Default 8300). This is used by servers to handle incoming requests from other agents. TCP only.
- Serf LAN (Default 8301). This is used to handle gossip in the LAN. Required by all agents. TCP and UDP.
- Serf WAN (Default 8302). This is used by servers to gossip over the WAN, to other servers. TCP and UDP. As of Consul 0.8 the WAN join flooding feature requires the Serf WAN port (TCP/UDP) to be listening on both WAN and LAN interfaces. See also: Consul 0.8.0 CHANGELOG and GH-3058
- HTTP API (Default 8500). This is used by clients to talk to the HTTP API. TCP only.
- DNS Interface (Default 8600). Used to resolve DNS queries. TCP and UDP.
还有一些配置参数这里就不一一介绍了,官方文档写的非常详细。
首先需要给环境变量 CONSUL_SERVER_IP_ADDR
赋值,若为 ECS 则为 ECS 的外网 IP,这里也是演示在 ECS 部署。
$ export CONSUL_SERVER_IP_ADDR={YOUR_IP_ADDR}
然后运行,这里没有使用 -d
参数,方便查看日志:
$ docker-compose -f docker-compose-server.yml up
运行 docker-compose 前,设置 CONSUL_SERVER_HOST
环境变量:
$ export CONSUL_SERVER_HOST={YOUR_SERVER_HOST}
然后运行:
$ docker-compose -f docker-compose-client.yml up
部署完成后可以通过 Consul UI 的 Nodes
看到两个 Healthy Nodes
:
也可以在部署好 Consul Client 的宿主机上运行:
$ consul members
这里要注意:
在查看了 Consul
仓库的 issue#1720 发现,Consul 提供的 Web UI 并没有提供 Auth 功能,所以可能要依靠第三方服务来实现,评论中也有提到:
highlyunavailable commented on 17 Feb 2016
When I ran a Consul Web UI I just used nginx and https://github.com/bitly/oauth2_proxy to provide authentication.
但是 HTTP 接口的权限,可以通过 Consul ACL 来控制,这是后话。
Consul 作为数据中心,提供了 k/v
存储的功能,我们可以利用这个功能为 Python 微服务提供配置中心。
Consul 提供了 HTTP 接口,我们可以从他的接口获取数据,当然我们不用自己去实现,python-consul 已经帮我们造好了轮子。
而且官方文档非常贴心,已经贴好了 Python 常用框架的一些 demo 代码:
from tornado.ioloop import IOLoop
from tornado.gen import coroutine
from consul.base import Timeout
from consul.tornado import Consul
class Config(object):
def __init__(self, loop):
self.foo = None
loop.add_callback(self.watch)
@coroutine
def watch(self):
c = Consul()
# asynchronously poll for updates
index = None
while True:
try:
index, data = yield c.kv.get('foo', index=index)
if data is not None:
self.foo = data['Value']
except Timeout:
# gracefully handle request timeout
pass
if __name__ == '__main__':
loop = IOLoop.instance()
_ = Config(loop)
loop.start()
Consul Template 提供一个方便的方式从 Consul 获取数据通过 consul-template 的后台程序保存到文件系统。
这个后台进程监控 Consul 示例的变化并更新任意数量的模板到文件系统.作为一个附件功能,模板更新完成后 consul-template 可以运行任何命令.可以查看示例部分看这个功能将会对哪些应用场景产生帮助。
首先需要在 Consul Client 所在的宿主机安装 consul-template
,由于 Demo 宿主机环境为 Mac OS,所以可以直接用 HomeBrew
进行安装。
$ brew install consul-template
安装完成后进入仓库的 python-web-service
路径,这是一个用 tornado 写的简单的 Web 服务。执行如下命令:
$ cd python-web-service && docker-compose up -d
等待命令运行完成,服务启动后,访问 localhost:8888
可以看到返回内容:
$ curl http://localhost:8888
Hello World
然后我们回到仓库路径,进入 consul-template
目录,该目录主要包含以下两个文件:
$ cd ../consul-template
$ tree
.
├── config.hcl # consul-template 配置文件
└── config.py.ctmpl # python-web-service 配置模版文件
查看一下 config.hcl
文件的内容:
consul {
address = "127.0.0.1:8500"
}
template {
source = "./config.py.ctmpl"
destination = "../python-web-service/config.py"
command = "docker restart python-web-service_python-web-service_1"
}
先介绍一下 *.hcl
配置文件,这个是 Consul 中非常常见的配置文件格式,也是 HashiCorp
下的产品所用的主要配置文件格式。配置文件中包含了 4
个重要的参数:
address
—— Consul Client 的访问地址和端口source
—— 需要配置的服务的配置文件模板destination
—— 配置文件渲染后输出的路径command
—— 当配置变更后,需要执行的命令再来看看模板文件 config.py.ctmpl
:
# -*- coding: utf-8 -*-
__author__ = 'gzp'
GREETING = '{{ keyOrDefault "python-web-service/greeting" "Hello World" }}'
模版文件的格式非常类似 Jinja2
的语法,这里的意思获取 key
为 python-web-service/greeting
下的值,默认值为 HelloWorld
。
接下来运行命令使 consul-template
生效:
$ consul-template -config config.hcl
我们可以访问 Consul Web UI 的 Key/Value
来修改我们的值:
将 Hello World
修改为 Hello Consul
,配置可能没有立即生效,若看到 consul-template
输出,则代表配置生效,服务以及重启:
$ consul-template -config config.hcl
python-web-service_python-web-service_1
然后再次访问一下 web 服务:
$ curl http://localhost:8888
Hello Consul
可以看到配置已经生效。