基于 Consul 的服务发现

Consul 是由 HashiCorp 开发的一个支持多数据中心的分布式服务发现和键值对存储服务的开源软件,被大量应用于基于微服务的软件架构当中。

Consul 初体验

用户可以通过 Consul 官网https://www.consul.io/downloads.html 下载对应操作系统版本的软件包。Consul 与 Prometheus 同样使用 Go 语言进行开发,因此安装和部署的方式也极为简单,解压并将命令行工具放到系统PATH路径下即可。

在本地可以使用开发者模式在本地快速启动一个单节点的 Consul 环境:

  1. $ consul agent -dev
  2. ==> Starting Consul agent...
  3. ==> Consul agent running!
  4. Version: 'v1.0.7'
  5. Node ID: 'd7b590ba-e2f8-3a82-e8a8-2a911bdf67d5'
  6. Node name: 'localhost'
  7. Datacenter: 'dc1' (Segment: '<all>')
  8. Server: true (Bootstrap: false)
  9. Client Addr: [127.0.0.1] (HTTP: 8500, HTTPS: -1, DNS: 8600)
  10. Cluster Addr: 127.0.0.1 (LAN: 8301, WAN: 8302)
  11. Encrypt: Gossip: false, TLS-Outgoing: false, TLS-Incoming: false

在启动成功后,在一个新的 terminal 窗口中运行 consul members 可以查看当前集群中的所有节点:

  1. $ consul members
  2. Node Address Status Type Build Protocol DC Segment
  3. localhost 127.0.0.1:8301 alive server 1.0.7 2 dc1 <all>

用户还可以通过HTTP API的方式查看当前集群中的节点信息:

  1. $ curl localhost:8500/v1/catalog/nodes
  2. [
  3. {
  4. "ID": "d7b590ba-e2f8-3a82-e8a8-2a911bdf67d5",
  5. "Node": "localhost",
  6. "Address": "127.0.0.1",
  7. "Datacenter": "dc1",
  8. "TaggedAddresses": {
  9. "lan": "127.0.0.1",
  10. "wan": "127.0.0.1"
  11. },
  12. "Meta": {
  13. "consul-network-segment": ""
  14. },
  15. "CreateIndex": 5,
  16. "ModifyIndex": 6
  17. }
  18. ]

Consul还提供了内置的DNS服务,可以通过Consul的DNS服务的方式访问其中的节点:

  1. $ dig @127.0.0.1 -p 8600 localhost.node.consul
  2. ; <<>> DiG 9.9.7-P3 <<>> @127.0.0.1 -p 8600 localhost.node.consul
  3. ; (1 server found)
  4. ;; global options: +cmd
  5. ;; Got answer:
  6. ;; ->>HEADER<<- opcode: QUERY, status: NOERROR, id: 50684
  7. ;; flags: qr aa rd; QUERY: 1, ANSWER: 1, AUTHORITY: 0, ADDITIONAL: 1
  8. ;; WARNING: recursion requested but not available
  9. ;; OPT PSEUDOSECTION:
  10. ; EDNS: version: 0, flags:; udp: 4096
  11. ;; QUESTION SECTION:
  12. ;localhost.node.consul. IN A
  13. ;; ANSWER SECTION:
  14. localhost.node.consul. 0 IN A 127.0.0.1
  15. ;; Query time: 5 msec
  16. ;; SERVER: 127.0.0.1#8600(127.0.0.1)
  17. ;; WHEN: Sun Apr 15 22:10:56 CST 2018
  18. ;; MSG SIZE rcvd: 66

在 Consul 当中服务可以通过服务定义文件或者是 HTTP API 的方式进行注册。这里使用服务定义文件的方式将本地运行的 node_exporter 通过服务的方式注册到 Consul 当中。

创建配置目录:

  1. sudo mkdir /etc/consul.d
  1. echo '{"service": {"name": "node_exporter", "tags": ["exporter"], "port": 9100}}' \
  2. | sudo tee /etc/consul.d/node_exporter.json

重新启动Consul服务,并且声明服务定义文件所在目录:

  1. $ consul agent -dev -config-dir=/etc/consul.d
  2. ==> Starting Consul agent...
  3. 2018/04/15 22:23:47 [DEBUG] agent: Service "node_exporter" in sync

一旦服务注册成功之后,用户就可以通过 DNS 或 HTTP API 的方式查询服务信息。默认情况下,所有的服务都可以使用 NAME.service.consul 域名的方式进行访问。

例如,可以使用 node_exporter.service.consul 域名查询 node_exporter 服务的信息:

  1. $ dig @127.0.0.1 -p 8600 node_exporter.service.consul
  2. ;; QUESTION SECTION:
  3. ;node_exporter.service.consul. IN A
  4. ;; ANSWER SECTION:
  5. node_exporter.service.consul. 0 IN A 127.0.0.1

如上所示DNS记录会返回当前可用的 node_exporter 服务实例的IP地址信息。

除了使用DNS的方式以外,Consul 还支持用户使用 HTTP API 的形式获取服务列表:

  1. $ curl http://localhost:8500/v1/catalog/service/node_exporter
  2. [
  3. {
  4. "ID": "e561b376-2c1b-653d-61a0-1d844bce06a7",
  5. "Node": "localhost",
  6. "Address": "127.0.0.1",
  7. "Datacenter": "dc1",
  8. "TaggedAddresses": {
  9. "lan": "127.0.0.1",
  10. "wan": "127.0.0.1"
  11. },
  12. "NodeMeta": {
  13. "consul-network-segment": ""
  14. },
  15. "ServiceID": "node_exporter",
  16. "ServiceName": "node_exporter",
  17. "ServiceTags": [
  18. "exporter"
  19. ],
  20. "ServiceAddress": "",
  21. "ServiceMeta": {},
  22. "ServicePort": 9100,
  23. "ServiceEnableTagOverride": false,
  24. "CreateIndex": 6,
  25. "ModifyIndex": 6
  26. }
  27. ]

Consul 也提供了一个 Web UI 可以查看 Consul 中所有服务以及节点的状态:

Consul UI

Consul UI

当然Consul还提供了更多的API用于支持对服务的生命周期管理(添加、删除、修改等)这里就不做过多的介绍,感兴趣的同学可以通过Consul官方文档了解更多的详细信息。

与 Prometheus 集成

Consul作为一个通用的服务发现和注册中心,记录并且管理了环境中所有服务的信息。Prometheus通过与Consul的交互可以获取到相应Exporter实例的访问信息。在Prometheus的配置文件当可以通过以下方式与Consul进行集成:

  1. - job_name: node_exporter
  2. metrics_path: /metrics
  3. scheme: http
  4. consul_sd_configs:
  5. - server: localhost:8500
  6. services:
  7. - node_exporter

在consul_sd_configs定义当中通过server定义了Consul服务的访问地址,services则定义了当前需要发现哪些类型服务实例的信息,这里限定了只获取node_exporter的服务实例信息。