idetcd is an etcd-based CoreDNS plugin used for identifying nodes in a cluster without domain name collision.
- idetcd uses same configuration on every node and asks nodes to dynamically config the cluster.
- idetcd can allow users easily take down/bring up new node without affecting other nodes in the cluster.
In the distributed system, identifying nodes in the cluster is a big challenge since it's quite often that nodes can get down or start/restart in the cluster which contains thousands of nodes and would be quite annoying if rebooting is needed after the membership in the cluster. For tackling this problem, usually requiring some complicated protocols or additional DevOps, what's more, in most case it needs to customize the configuration for every different node which requires lots of work and also adds some risks to manage the system.
Distributed TensorFlow is also encountering the similar problems[1]. In the older version of distributed TensorFlow, adding a node to the cluster is not that easy. First of all, you need to bring the whole system down, and then customize a configuration for the new node, after that restart the whole system. It did require additional DevOps work and it's also not "friendly" for some machine learning lovers to set up their own distributed TensorFlow clusters because, for people who are not familiar with distributed system, it would be great if she/he can add/delete a node from the system by one or two commands.
In practice, there are some approaches to solve this problem. For example, as mentioned before, building some protocols on the top of current Tensorflow codebase definitely can achieve this goal, but it probably is not a good way since may need to change the structure of Tensorflow, and bring some unnecessary complexities. And also, people who are not familiar with the infrastructure of TensorFlow still can not do anything if they still meet some problems when they deploy their own clusters. A more flexible way to do this is adding a separate module like DNS server, and nodes expose themselves through DNS.
CoreDNS has a plugin-based architecture and it is a really lightweight, flexible and extendable DNS server which can easily enable the customized plugin. For solving this issue, we can set up the CoreDNS plus customized plugin on every node in the TensorFlow cluster, and use the plugin to write/read DNS records in a distributed key-value store, like zookeeper and etcd. And this is what idetcd does.
The figure[2] above shows the scenario of how it works. The idea is quite simple: Set up CoreDNS server on every node, and node exposes itself by taking the free domain name.
In details, before the cluster is started, we set up CoreDNS server on every node in the cluster, for every node we just use the same configuration(See [example] below for details)which specifies the domain name pattern of nodes in this cluster, like worker{{.ID}}.tf.local., the maximum number of node allowed in this cluster, and etcd endpoints. Then we just start up all the nodes. That's it!
Notice, at the starting time, all the nodes haven't exposed themselves to other nodes. Then we just start CoreDNS server on every node, and nodes will try to find free slots in the etcd to expose. For example, the node may first try to take worker1.tf.local., and then it will try to figure out whether this domain name already exists in the etcd: if the answer is yes, then the node will try to increase the id to 2 and look into etcd again; otherwise, it will just take the name, and write it to the etcd. In this way, every node can dynamically find a domain name for itself without any collision. And also we don't need to customize the configuration for every node; instead, we use the same configuration and let the nodes expose themselves!
CoreDNS uses a configuration file called Corefile to specify the configuration, please go to CoreDNS Github repo for more details. Here is a snippet for idetcd syntax:
idetcd {
endpoint ENDPOINT...
limit LIMIT
pattern PATTERN
}
endpoint
ENDPOINT the etcd endpoints. Defaults to "http://localhost:2379".limit
LIMIT the maximum limit of the node number in the cluster, if some nodes is going to expose itself after the node number in the cluster hits this limit, it will fail.pattern
PATTERN the domain name pattern that every node follows in the cluster. And here we use golang template for the pattern.
In the following example, we are going to start up a cluster which contains 5 nodes, on every node we can get this project by:
$ go get -u github.com/jiachengxu/idetcd
Before you move to the next step, make sure that you've already set up a etcd instance, and don't forget to write down the endpoints.
Then you need to add a Corefile which specifys the configuration of the CoreDNS server in the same directory of main.go
, a simple Corefile example is as follows, please go to CoreDNS Github repo for more details.
. {
idetcd {
endpoint ETCDENDPOINTS
limit 5
pattern worker{{.ID}}.tf.local.
}
}
And then you can generate binary file by:
$ go build -v -o coredns
Alternatively, if you have docker installed, you could also execute the following to build:
$ docker run --rm -i -t -v $PWD:/go/src/github.com/jiachengxu/idetcd \
-w /go/src/github.com/jiachengxu/idetcd golang:1.10 go build -v -o coredns
Then run it by:
$ ./coredns
After that, all nodes in the cluster are trying to find free slots in the etcd to expose themselves, once they succeed, you can get the domain name of every node on every node in the same cluster by:
$ dig +short worker4.tf.local @localhost
Also ipv6 is supported:
$ dig +short worker4.tf.local AAAA @localhost
Using CoreDNS with idetcd plugin to config the cluster is a one-time process which is different with the general config process. For example, if you want to set up a cluster which contains several instances on AWS, you can use the same configuration for every instance and let all the instances to expose themselves in the init
process. This can be achieved by using cloud-init
in user data
. Here is a bash script example for AWS instances to execute at launch:
#!/bin/bash
set -x
## Install docker.
yum install -y docker
echo
chkconfig docker on
service docker start
echo
## Install git.
yum install -y git
git clone https://github.com/jiachengxu/idetcd.git /home/ec2-user/idetcd
cd /home/ec2-user/idetcd
## Using docker to build the binary file of CoreDns with idetcd plugin specified.
docker run --rm -v $PWD:/go/src/github.com/jiachengxu/idetcd -w /go/src/github.com/jiachengxu/idetcd golang:1.10 go build -v -o coredns
## Create a Corefile for specifying the configuration of CoreDNS.(Don't forget to replace the ETCDENDPOINTS and NUMBER with your own etcd endpoints and limit of node in the cluster!)
cat > Corefile << EOF
. {
idetcd {
endpoint ETCDENDPOINTS
limit NUMBER
pattern worker{{.ID}}.tf.local.
}
}
EOF
./coredns
[1]Dynamic RPC Address Resolution
[2]Lightning Talk: Scaling Distributed Deep Learning with Service Discovery: How CoreDNS Helps Distributed TensorFlow Tasks - Yong Tang, Infoblox Inc.