设为首页 收藏本站
查看: 1179|回复: 0

[经验分享] ZooKeeper场景实践:(2)集中式配置管理

[复制链接]

尚未签到

发表于 2015-11-21 16:05:03 | 显示全部楼层 |阅读模式

1. 基本介绍



在分布式的环境中,可能会有多个对等的程序读取同样的配置文件,程序可以部署在多台机器上,如果配置采用文件的话,则需要为部署该程序的机器也部署一个配置文件,一旦要修改配置的时候就会非常麻烦,需要修改多个配置文件,而且容易产生不一致。

集中式配置管理的思路是,将配置数据集中发布到ZooKeeper的节点上,供订阅者动态获取数据。实现配置的集中式管理和动态更新。可以简单的理解为配置数据与程序分离。


2. 场景分析



(1).集中式配置管理



通常来说,大部分项目里面都有约定的配置文件格式,如ini,xml等。一般都会有对应的解析库类。这种解析库类的基本工作模式为:


  • 读取文件(open)
  • 解析文件(parse)
  • 对外提供参数(get)


如果我们将文件的内容全部放到ZooKeeper的某个节点上.解析类将配置数据全部下载到本地,在完成解析的话,则可以用很小的改动就完成集中式配置管理的需求。


  • 读取Zookeeper上对应路径的数据(read)
  • 解析文件(parse)
  • 对外提供参数(get)


(2).动态更新



动态更新是希望不重启程序就能够实时获取更新的配置。在单机环境中,这种配置数据通常会放在数据库中,修改配置只需要update数据库就可以了。

使用ZooKeeper的话,需要节点注册一个watcher,监视配置数据的是否有变化,一定出现变化,则调用新的解析类来重新解析配置数据。

个人认为这个特征使用Zookeeper可以实现,但是并不是所有配置都需要这个功能,这种比较适合对配置敏感,需要实时更新配置的情况。


3. 动手实践



这里我只实现了集中式配置管理的功能,没有实现动态更新,有需要的话你可以尝试自己实现。

由于之前曾经做个一个ini文件的库类解析,这里就直接拿过来改了。根据场景的分析,只需要修改open这个函数就ok了。

看下原来的open函数

/*读取文件名要改为地址和路径*/
int IniFile::open(const string &filename)
{   
release();
fname_ = filename;
IniSection *section = NULL;
/*读取数据的方式需要修改*/
FILE *fp = fopen(filename.c_str(),"r");
if(fp == NULL ){
return -1;
}
string line;
string comment;
//增加默认段
section = new IniSection();
sections_[""] = section;
/*获取行的方式需要修改*/
while(getline(line,fp) > 0){
...//省略单行的解析
}
fclose(fp);
return 0;
}





我们有三个主要需要修改的地方,分别是是入参,fopen和getline。下面是修改后的open函数

/*修改入参,host为Zookeeper的ip及端口地址,filepath为配置数据的路径*/
int IniFile::open2(const string &host,const string &filepath)
{   
release();
fname_ = filepath;
IniSection *section = NULL;
char fp[2048]={0};
/*ZooKeeper来读取*/
zkopen(host,filepath,fp,sizeof(fp));
if(fp[0] == 0){
return -1;
}
string line;
string comment;
//增加默认段
section = new IniSection();
sections_[""] = section;
char *p = fp;
/*调整getline的入参*/
while(getline2(line,p) > 0){
...//省略单行的解析
}

return 0;
}





zkopen从Zookeeper的节点上读取数据,并保存到fp中。代码如下:

string zkopen(const string &host,const string &filepath,char *fp,int len)
{
int timeout = 30000;  
char path_buffer[512];  
int bufferlen=sizeof(path_buffer);
char conf_data[2048];
int conf_len=sizeof(conf_data);
zoo_set_debug_level(ZOO_LOG_LEVEL_WARN); //设置日志级别,避免出现一些其他信息  
zhandle_t* zkhandle = zookeeper_init(host.c_str(),NULL, timeout, 0, (char *)"Monitor Test", 0);  
if (zkhandle ==NULL)  
{  
fprintf(stderr, "Error when connecting to zookeeper servers...\n");  
exit(EXIT_FAILURE);  
}  
int ret = zoo_get(zkhandle,filepath.c_str(),0,conf_data,&conf_len,NULL);
if(ret != ZOK){
fprintf(stderr,"failed to get the data of path %s!\n",filepath.c_str());
conf_data[0] = 0;
}
zookeeper_close(zkhandle);
strncpy(fp,conf_data,len);
return conf_data;
}





接下来在对比下调用的变化。

原来的调用方式:

  /** read test **/
IniFile ini;
ini.open(g_filepath);
//获取指定段的指定项的值
int ret = 0;
string db_name = ini.getStringValue("COMMON","DB",ret);
string db_passwd = ini.getStringValue("COMMON","PASSWD",ret);





现在的调用方式:

/** read test **/
IniFile ini;
ini.open2(g_host,g_filepath);/*仅此处有变化*/
//获取指定段的指定项的值
int ret = 0;
string db_name = ini.getStringValue("COMMON","DB",ret);
string db_passwd = ini.getStringValue("COMMON","PASSWD",ret);





由上可见,配置的改造还是很容易的,而且对程序的改动很小。

代码详见https://github.com/Winnerhust/ZooKeeper-Exam/tree/master/Config


5.小提示



需要注意一点,配置文件中通常有很多换行,而ZooKeeper的客户端命令行工作不支持字符转义。比如你要将一个配置文件test.ini的内容保存到Zookeeper上,文件内容如下。

[COMMON]

DB=mysql

PASSWD=root

你可能会在Zookeeper客户端上输入:

[zk: 172.17.0.36:2181(CONNECTED) 39] create /Conf/test.ini [COMMON]\nDB=mysql\nPASSWD=root\n

结果与我们希望的并不一样:

[zk: 172.17.0.36:2181(CONNECTED) 43] get /Conf/test.ini3[COMMON]\nDB=mysql\nPASSWD=root\n

Zookeeper并没有将字符串进行转义,所以不能用ZooKeeper客户端直接上传配置文件。因此在代码里我还增加了一个上传配置的功能。只需要将上一个参数-r就可以了。如将test.ini文件的内容上传到ZooKeeper:

cat test.ini | testcase -r -p/Conf/test.ini -s172.17.0.36:2181

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-141928-1-1.html 上篇帖子: ZooKeeper基本理解 下篇帖子: ZooKeeper 节点类型
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表