利用MongoC自动化配置复制集(replSet)

1 篇文章 0 订阅
1 篇文章 0 订阅

MongoDB复制集简介

Mongodb复制集由一组Mongod实例(进程)组成,包含一个Primary节点和多个Secondary节点,Mongodb Driver(客户端)的所有数据都写入Primary,Secondary从Primary同步写入的数据,以保持复制集内所有成员存储相同的数据集,提供数据的高可用。
常见复制集架构

MongoDB C Driver

mongoc 是官方的mongo驱动,用于通过程序来控制mongo数据库。

目前所有MongoDB复制集的配置都是通过mongo客户端进行配置的,无法实现自动配置。
#传统的老方法,需要手动配置,无法自动化配置
mongo --port=28017
use admin
config = {_id: 'my_repl', members: [
                          {_id: 0, host: '10.0.0.134:28017'},
                          {_id: 1, host: '10.0.0.134:28018'},
                          {_id: 2, host: '10.0.0.134:28019'}]
          }
rs.initiate(config)
use test;
db.movies.insert([ { "title" : "Jaws", "year" : 1975, "imdb_rating" : 8.1 },{ "title" : "Batman", "year" : 1989, "imdb_rating" : 7.6 },
] );

现在新解决方法是通过编写后台程序MongoControlServer通过mongoc向mongo数据发送创建配置复制集命令。
架构

MongoC编写Command接口

//头文件
class mongodb_api
{
public:
	static mongodb_api& instance();
	/*MongoDB数据库初始化*/
	int mongodb_init(const char* url);
	/*清理数据库*/
	int mongodb_cleanup();
	int mongodb_do_command(const char* databasename, std::string command,std::string& replys);
private:
	mongodb_api();
	static mongodb_api* s_instance;
	void* m_mongodb_ptr=nullptr;
};
class  mango_db
{
public:
	int mongodb_init(const char* url);
	
	int create_client_pool();
	
	int destory_client_pool();
	//mongodb有一系列的自有的命令,包括管理类命令、地理信息命令等
	int  do_command(const char*databasename, const bson_t *command, bson_t &reply);
private:
	int m_ulthreads = 10;
	void *  m_pool=nullptr;
	void *  m_url = nullptr;
	std::string  m_strurl;
};
//mongoc的接口用于与mongo通信
#include <mongoc.h>
mongodb_api* mongodb_api::s_instance = nullptr;
mongodb_api& mongodb_api::instance()
{
	if (s_instance == nullptr)
	{
		s_instance = new mongodb_api;
	}
	return *s_instance;
}

int mango_db::mongodb_init(const char* url)
{
	int ulret = 0;
	m_strurl = url;
	mongoc_init();
	return ulret;
}
int mongodb_api::mongodb_cleanup()
{
	int ulret = 0;
	mango_db* dbapi = static_cast<h3c_cloud_mango_db*>(m_mongodb_ptr);
	if (dbapi == nullptr){ return ulret;}
	dbapi->destory_client_pool();
}
	
int mango_db::create_client_pool()
{
	int ulret = 0;
	mongoc_client_pool_t * client_pool=nullptr;
	mongoc_uri_t         *uri=nullptr;
	uri = mongoc_uri_new(m_strurl.c_str());//url指定ip地址,没有指定
	client_pool = mongoc_client_pool_new(uri);
	m_pool = static_cast<void*>(client_pool);
	m_url = static_cast<void*>(uri);
	return ulret;
}

int mango_db::destory_client_pool()
{
	int ulret = ERROR_COMMON_SUCCEED;
	if (m_url == nullptr || m_pool == nullptr)
	{
		return 1;
	}
	mongoc_client_pool_destroy(static_cast<mongoc_client_pool_t*>(m_pool));
	mongoc_uri_destroy(static_cast<mongoc_uri_t*>(m_url));
	mongoc_cleanup();
	return ulret;
}

int mongodb_api::mongodb_do_command(const char* databasename, std::string command, std::string& replys)
{
	int ulret = 0;
	mango_db* dbapi = static_cast<mango_db*>(m_mongodb_ptr);
	if (dbapi == nullptr){ return ERROR_COMMON_FALT; }
	bson_error_t errorinfo;
	bson_t reply;
	bson_t* cond = bson_new_from_json((const unsigned char*)command.c_str(), command.length(), &errorinfo);
	ulret = dbapi->do_command(databasename, cond, reply);
	char *condStr = bson_as_json(&reply, NULL);
	replys = condStr;
	bson_free(condStr);
	bson_destroy(&reply);
	return ulret;
}

int mango_db::do_command(const char*databasename,const bson_t *command, bson_t &reply)
{
	int ulret = 0;
	mongoc_client_pool_t *pool = static_cast<mongoc_client_pool_t*>(m_pool);
	mongoc_client_t      *client;
	bson_error_t error;
	client = mongoc_client_pool_pop(pool);//从连接池中获取连接对象 
	if (client == nullptr){
		H3C_CLOUD_MONGODB_LOG1(H3C_LOG_ERROR, "fail get client ptr url:[%s] database name:[%s]", databasename);
		return 1;
	}
	//删除指定的行
	if (!mongoc_client_command_simple(client, databasename, command, NULL, &reply, &error))
	{
		char *commandStr = bson_as_json(command, NULL);
		H3C_CLOUD_MONGODB_LOG4(H3C_LOG_ERROR, "fail do command url:[%s] database name:[%s]  command [%s] error [%s]", m_strurl.c_str(), databasename, commandStr, error.message);
		bson_free(commandStr);
		mongoc_client_pool_push(pool, client);
		return 1;
	}
	mongoc_client_pool_push(pool, client);
	return 0;
}

通过MongoC编写复制集的配置

以下是编写的测试代码,用于添加复制集节点。生产环境下可以编写一个后台程序自动检测mongo后台服务,自己检查mongo ip 自动生成**"{“replSetInitiate”:{"_id":“COOL”,“members”:[{"_id":0,“host”:“10.90.15.198:27017”},{"_id":1,“host”:“10.90.15.199:27017”},{"_id":2,“arbiterOnly”:true,“host”:“10.90.15.200:27017”}]}}"**; 配置文件,可以实现mongo数据库的自动更新,创建复制集(replSet)

//通过mongoc进行复制集配置(例子)
ULONG_32 h3c_cloud_manager_db::h3c_cloud_command()
{
	ULONG_32 ulret = ERROR_COMMON_SUCCEED;
	//检查复制集
	std::string command = "{\"replSetGetStatus\":1}";
	std::string reply;
	ulret = h3c_cloud_mongodb_api::instance().mongodb_do_command("admin", command,reply);
	if (ulret != ERROR_COMMON_SUCCEED){
		H3C_CLOUD_MAN_LOG2(H3C_LOG_ERROR, "fail create json  info [%s] [%s]", command.c_str(),reply.c_str());
	}
	//添加复制节点
	
	std::string command = "{\"replSetInitiate\":{\"_id\":\"COOL\",\"members\":[{\"_id\":0,\"host\":\"10.90.15.198:27017\"},{\"_id\":1,\"host\":\"10.90.15.199:27017\"},{\"_id\":2,\"arbiterOnly\":true,\"host\":\"10.90.15.200:27017\"}]}}";
	std::string reply;
	ulret = h3c_cloud_mongodb_api::instance().mongodb_do_command("admin", command, reply);
	if (ulret != ERROR_COMMON_SUCCEED){
		H3C_CLOUD_MAN_LOG2(H3C_LOG_ERROR, "fail create json  info [%s] [%s]", command.c_str(), reply1.c_str());
	}
	//再次检查
	command = "{\"replSetGetStatus\":1}";
	reply;
	ulret = h3c_cloud_mongodb_api::instance().mongodb_do_command("admin", command, reply);
	if (ulret != ERROR_COMMON_SUCCEED){
		H3C_CLOUD_MAN_LOG2(H3C_LOG_ERROR, "fail create json  info [%s] [%s]", command.c_str(), reply2.c_str());
	}
	return ulret;
}
#上面测试接口三个命令返回的参数
{ "info" : "run rs.initiate(...) if not yet done for the set", "ok" : 0.0, "errmsg" : "no replset config has been received", "code" : 94, "codeName" : "NotYetInitialized" }

{ "ok" : 1.0 }

{ "set" : "COOL", "date" : { "$date" : 1573528438228 }, "myState" : 2, "term" : 0, "heartbeatIntervalMillis" : 2000, "optimes" : { "lastCommittedOpTime" : { "ts" : { "$timestamp" : { "t" : 0, "i" : 0 } }, "t" : -1 }, "appliedOpTime" : { "ts" : { "$timestamp" : { "t" : 1573528434, "i" : 1 } }, "t" : -1 }, "durableOpTime" : { "ts" : { "$timestamp" : { "t" : 1573528434, "i" : 1 } }, "t" : -1 } }, "members" : [ { "_id" : 0, "name" : "10.90.15.198:27017", "health" : 1.0, "state" : 2, "stateStr" : "SECONDARY", "uptime" : 73, "optime" : { "ts" : { "$timestamp" : { "t" : 1573528434, "i" : 1 } }, "t" : -1 }, "optimeDate" : { "$date" : 1573528434000 }, "infoMessage" : "could not find member to sync from", "configVersion" : 1, "self" : true }, { "_id" : 1, "name" : "10.90.15.199:27017", "health" : 1.0, "state" : 2, "stateStr" : "SECONDARY", "uptime" : 4, "optime" : { "ts" : { "$timestamp" : { "t" : 1573528434, "i" : 1 } }, "t" : -1 }, "optimeDurable" : { "ts" : { "$timestamp" : { "t" : 1573528434, "i" : 1 } }, "t" : -1 }, "optimeDate" : { "$date" : 1573528434000 }, "optimeDurableDate" : { "$date" : 1573528434000 }, "lastHeartbeat" : { "$date" : 1573528438080 }, "lastHeartbeatRecv" : { "$date" : 1573528437780 }, "pingMs" : 1, "configVersion" : 1 }, { "_id" : 2, "name" : "10.90.15.200:27017", "health" : 1.0, "state" : 7, "stateStr" : "ARBITER", "uptime" : 4, "lastHeartbeat" : { "$date" : 1573528438080 }, "lastHeartbeatRecv" : { "$date" : 1573528438173 }, "pingMs" : 0, "configVersion" : 1 } ], "ok" : 1.0 }

注:接口配置的详细参数

//std::string command 的详细配置-->用于这个命令的cfg参数{\"replSetInitiate\":\"cfg\"}
{
	"_id" : "COOL",
	"version" : 1,
	"protocolVersion" : NumberLong(1),
	"members" : [
		{
			"_id" : 0,
			"host" : "10.90.15.198:27017",
			"arbiterOnly" : false,
			"buildIndexes" : true,
			"hidden" : false,
			"priority" : 1,
			"tags" : {
			},
			"slaveDelay" : NumberLong(0),
			"votes" : 1
		},
		{
			"_id" : 1,
			"host" : "10.90.15.199:27017",
			"arbiterOnly" : false,
			"buildIndexes" : true,
			"hidden" : false,
			"priority" : 1,
			"tags" : {
			},
			"slaveDelay" : NumberLong(0),
			"votes" : 1
		},
		{
			"_id" : 2,
			"host" : "10.90.15.200:27017",
			"arbiterOnly" : false,
			"buildIndexes" : true,
			"hidden" : false,
			"priority" : 1,
			"tags" : {
			},
			"slaveDelay" : NumberLong(0),
			"votes" : 1
		}
	],
}

mongodb.conf配置文件

#mongodb.conf mongo数据库配置文件
# 是否启用认证
auth= true
# 集群的私钥的完整路径,只对于Replica Set 架构有效(noauth = true时不用配置此项)

#在一台服务器上生成私钥
# openssl rand -base64 745 > mongodb-keyfile
#修改私钥的权限为可读写
#chmod 600 mongodb-keyfile
#将私钥拷贝到其余的服务器上
#scp mongodb-keyfile root@xxx.xxx.xxx.xxx:/xxxx/configdb 
keyFile = /data/configdb/mongodb-keyfile 

# 设置oplog的大小(MB)
oplogSize=2048
# 设置副本集名称
replSet=COOL
# 绑定ip
bind_ip = 0.0.0.0
# 监听端口
port=27017 

mongo docker-compose 模板文件,这里使用docker 安装mongo

# Use root/example as user/password credentials
version: '3.1'

services:
  mongo:
    image: mongo
    restart: always
    privileged: true
    ports:
      - 27017:27017
    volumes:
      - ../data:/data/db
      - ../config:/data/configdb
     
    command: mongod -maxConns=4000 --config /data/configdb/mongodb.conf
      
  mongo-express:
    image: mongo-express
    networks:
      #- default
      - cloudface_net
    restart: always
    ports:
      - 27109:8081
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值