ETCD
更新/删除/重新设置键的ttl 都会触发watcher , 但是如果在body中增加refresh=true , 更新ttl(必须存在) , 将不会出发watcher事件。
?wait=true 监听当前节点
?recursive=true 递归监听当前节点和子目录
?waitIndex=xxx 监听过去已经发生的。过去值的查询或监听, 必选与wait一起使用。
watch 一个ttl自删除的key时,收到如下 “expire” action。
{
"action": "expire",
"node": {
"createdIndex": 2223,
"key": "/message",
"modifiedIndex": 2224
},
"prevNode": {
"createdIndex": 2223,
"expiration": "2018-11-12T09:25:00.028597482Z",
"key": "/message",
"modifiedIndex": 2223,
"value": ""
}
}
GET 对过去的键值操作进行查询:类似上面提到的监控,在其基础上指定过去某次修改的索引编号,就可以查询历史操作。默认可查询的历史记录为1000条。
?waitIndex=xxx 监听过去已经发生的。 这个在确保在watch命令中,没有丢失事件非常有用。例如:我们反复watch 我们得到节点的 modifiedIndex+1。
因为 node的modifiedIndex的值是不连续,如果waitIndex的值没有相应modifiedIndex,返回比它大的最近的modifedIndex的节点信息。 如果大于节点中所有的modifiedIndex,等待,直到节点的modifiedIndex值大于等于waitIndex的值。
即使删除key后,也可以查询历史数据。
store中有一个全局的currentIndex,每次变更,index会加1.然后每个event都会关联到currentIndex.
当客户端调用watch接口(参数中增加 wait参数)时,如果监听的waitIndex 不存在与key对应的EventHistroy 中(currentIndex >= waitIndex),并且key对应的modifyIndex > waitIndex , 则会查找到第一个大于waitIndex 的数据进行展示。
如果历史表中没有或者请求没有带 waitIndex,则放入WatchHub中,每个key会关联一个watcher列表。 当有变更操作时,变更生成的event会放入EventHistroy表中,同时通知和该key相关的watcher。
注意:
1. 必须与 wait 一起使用;
2. curl 中url需要使用引号。
3. etcd 仅仅保留系统中所有key最近的1000条event,建议将获取到的response发送到另一个线程处理,而不是处理response而阻塞watch。
4. 如果watch超出了etcd保存的最近1000条,建议get后使用response header中的 X-Etcd-Index + 1进行重新watch,而不是使用node中的modifiedIndex+1. 因为 X-Etcd-Index 永远大于等于modifiedIndex, 使用modifiedIndex可能会返回401错误码,同样超出。
5. long polling可能会被服务器关闭,如超时或服务器关闭。导致仅仅收到header 200OK,body为空,此时应重新watch。
etcd4j 测试案例:
import mousio.etcd4j.EtcdClient;
import mousio.etcd4j.promises.EtcdResponsePromise;
import mousio.etcd4j.responses.EtcdKeysResponse;
import org.junit.Before;
import org.junit.Test;
import java.io.IOException;
import java.net.URI;
import java.util.List;
/**
* @Description
* @auther bozhu
* @create 11\ 12\ 2018
*/
public class EtcdClientAddListenerTest {
EtcdClient client = null;
@Before
public void executeBefore() {
client = new EtcdClient(URI.create("http://172.16.85.20:2379"));
}
private String getConfig(String configFile , EtcdKeysResponse dataTree) {
List<EtcdKeysResponse.EtcdNode> nodes = null;
if(null != dataTree ) {
return dataTree.getNode().getValue();
}
System.out.println("Etcd configFile"+ configFile+"is not exist,Please Check");
return null;
}
@Test
public void testStask() throws Exception {
for (int i = 1; i < 5000; i++) {
EtcdKeysResponse etcdKeysResponse = client.put("/xdriver/test/value", "" + i).send().get();
System.out.println(etcdKeysResponse);
}
}
@Test
public void testListener() throws Exception{
this.startListenerThread(client , "/xdriver/test/value");
Thread.sleep(1000000L);
}
public void startListenerThread(EtcdClient client , String dir) throws Exception{
EtcdKeysResponse etcdKeysResponse = client.get(dir).send().get();
System.out.println(etcdKeysResponse.getNode().getValue());
new Thread(()->{
startListener(client,dir,etcdKeysResponse.getNode().getModifiedIndex() + 1);
}).start();
}
public void startListener(final EtcdClient client , final String dir , long waitIndex) {
EtcdResponsePromise<EtcdKeysResponse> promise = null;
try {
// 如果监听的waitIndex 不存在与key对应的EventHistroy 中(currentIndex >= waitIndex) ,
// 并且key对应的modifyIndex > waitIndex , 则会查找到第一个大于waitIndex 的数据进行展示
promise = client.get(dir).waitForChange(waitIndex).consistent().send();
} catch (IOException e) {
e.printStackTrace();
}
promise.addListener(promisea -> {
try {
EtcdKeysResponse etcdKeysResponse = promisea.get();
new Thread(() -> {startListener(client , dir , etcdKeysResponse.getNode().getModifiedIndex() + 1);}).start();
String config = getConfig(dir, etcdKeysResponse);//加载配置项
System.out.println(config);
} catch (Exception e) {
e.printStackTrace();
System.out.println("listen etcd 's config change cause exception:{}"+e.getMessage());
}
});
}
}