/cloriConf

A format-independent C++ configuration library with support for both ini, json, yaml style config file and zookeeper/etcd.(配置解析库&轻量级配置中心)

Primary LanguageC++MIT LicenseMIT

中文版

cloriConf

Suppose you are programing in C++ and looking for a third-party configuration library, you may considering cloriConf, which has characters of both lightweight configuration library and distributed configuration center.

Features

The most significant features that cloriConf exceeds general configuration libraries are

  • Format-independent - As designed, cloriConf's access APIs are completely separated from specific configuration format. Actually cloriConf trys to translate configuration items of any config format(e.g. ini, json, yaml) into nodes of a configuration tree, and what cloriConf's API need to do is just searching a configuration tree.The currently supported formats are json, joml((James's Obvious Minimal Language, a superset of INI format, see About JOML for detail), and more config format will be supported in the future.
  • Sourece-independent - Not only can cloriConf load various configuration format from local file and read config from a string directly, but also accessing zookeeper(support for ectd will come soon) is OK.

Multifunctional though cloriConf is, cloriConf's access API is designed to be as simple as a lightweight configuration library, Refer Usage and API Reference for details.

Usage

The following will take you through how to use cloriConf in your program.Or you can refer API Reference for detail of cloriConf APIs.

  • Access INI-style(JOML-style) configuration file
    common.ini:
    [[adslot=xxx]]
      splash=123
      popUp=345
      [vta]
      splash=222 #comment test ; semicolon test
      popUp=3415

C++ code:

    #include <cloriconf/config.h>
    
    // SRC_LOCAL: load config from local file
    // FMT_JOML: parse as INI-style(JOML-style) config 
    // CMT_SHARP: take '#' as line comment identifier
    Config* conf = Config::instance()->Load("../conf/common.ini", SRC_LOCAL | FMT_JOML | CMT_SHARP);
    if (conf) {
        std::cout << "parse ini config file success" << std::endl;
    } else {
        std::cout << "parse ini config file failed"; 
        return;
    }
    std::string val1 = conf->GetString("adslot.vta.popUp");
    ConfNode* node = conf->GetConfNode("adslot.vta.splash");
    if (node) {
        std::cout << "adslot.vta.splash=" << node->AsString() << std::endl;
    }
    int val2 = conf->GetInt32("/adslot/vta/splash");
    std::cout << "val1=" << val1 << std::endl;
    std::cout << "val2=" << val2 << std::endl;
  • Access JSON-style configuration file
    common.json:
    {
        "adslot":{
            "splash":"123",
            "popUp":"345",
            "vta":{
                "splash":"222",
                "popUp":"3415"
            }   
        }   
    }

C++ code:

    #include <cloriconf/config.h>

    // FMT_JSON: parse as JSON-style config
    std::string err_msg;
    Config* conf = Config::instance()->Load("../conf/common.json", SRC_LOCAL | FMT_JSON, &err_msg);
    if (conf) {
        std::cout << "parse json config file success" << std::endl;
    } else {
        std::cout << "parse json config file failed, " << err_msg << std::endl;
    }   
    std::string val1 = conf->GetString("adslot.vta.popUp");
    const ConfNode* node = conf->GetConfNode("adslot.vta.splash");
    if (node) {
        std::cout << "adslot.vta.splash=" << node->AsString() << std::endl; // 222
        std::cout << "adslot.vta.popUp=" << conf->GetString("adslot.vta.popUp") << std::endl; // 3415
    }   

    // take 13456789 as default value when node "/adslot/vta/not_exist" not found
    int val2 = conf->GetInt32("/adslot/vta/not_exist", 13456789);
    std::cout << "val1=" << val1 << std::endl;
    std::cout << "val2=" << val2 << std::endl;
  • Load config from zookeeper
    zk.ini(JOML-style):
    [zookeeper]
        # host=10.6.43.15:2181,10.6.43.16:2181,10.6.1.12:2181,10.6.1.13:2181
        host=localhost:2181
        timeout=3000
        interval=5000
        root=/online/commercial/ssp
    []

C++ code:

    #include <cloriconf/config.h>

    // SRC_ZK: load config from zookeeper
    std::string err_msg;
    Config* conf = Config::instance()->Load("../conf/zk.ini", SRC_ZK, &err_msg);
    if (conf) {
        std::cout << "run test success" << std::endl;
    } else {
        std::cout << "run test failed, " << err_msg << std::endl;
        return;
    }
    std::string val1 = conf->GetString("rules/popUp");
    std::cout << "value of rules/popUp=" + val1 << std::endl;

    ConfNode* node = conf->GetConfNode("rules/splash");
    if (node) {
        for (ConfNode::ChildrenIterator iter = node->begin(); iter != node->end(); ++iter) {
            std::cout << "zookeeper config node, key=" << iter->name() << ", value=" << iter->AsString() << std::endl;
        }   
    }

Installation

Before installation, you have to confirm that

  • CloriConf currently has passed the test in Linux operation system only. To support other OS, you may need to modify CMakeLists.txt and do some other coding work.
  • CloriConf is not a fully self-contained library, which has dependency on RapidJSON(for json support) and zookeeper(for zookeeper support). To simplify the installation of cloriConf, function for json/zookeeper parser is disabled by default.
    To install cloriConf, you can run the following command in root path of cloriConf source code:
// (without support for JSON-style and zookeeper parser) 
mkdir build && cd build
cmake ..
make
sudo make install

To support JSON-style parser, you need to install RapidJSON firstly, and then run

mkdir build && cd build
cmake .. -DENABLE_JSON=ON
make
sudo make install

To support zookeeper, you need to install zookeeper firstly, and then run

mkdir build && cd build
cmake .. -DENABLE_ZOOKEEPER=ON
make
sudo make install

Alternatively you can specify the install prefix by setting CMAKE_INSTALL_PREFIX, in this case, a complete step seems like

mkdir build && cd build
cmake .. -DENABLE_JSON=ON -DENABLE_ZOOKEEPER=ON -DCMAKE_INSTALL_PREFIX=/usr/local/third_party
make
sudo make install

After adding cloriConf to your program, you can compile like (assume cloriConf installed in /home/weijian/cloriconf)

g++ tutorial.cc -I/home/weijian/cloriconf/include -L/home/weijian/cloriconf/lib -lcloriconf -o main -std=c++11 -Wl,-rpath=/home/weijian/cloriconf/lib

Build Up Zookeeper Dashboard by CloriConf

As cloriConf support loading config data from zookeeper, you can use it as a simple distributed configuration center to some extent. What you need is a zookeeper cluster.
To simplify zookeeper management, cloriConf contain a zookeeper dashboard module in directory src/dashboard. To use cloriConf dashboard, a PHP runtime environment and nginx are required, and then you can add the following nginx configuration into nginx.conf and restart nginx

    # set "/home/weijian/github/cloriConf" to your own directory
    location ~ \.php$ {
        root    /home/weijian/github/cloriConf/src/dashboard;
        fastcgi_pass   127.0.0.1:9000;
        fastcgi_index  index.php;
        fastcgi_param  SCRIPT_FILENAME  $document_root$fastcgi_script_name;
        include        fastcgi_params;
    }   
    location ~ \.(css|js|html)$ {
        root        /home/weijian/github/cloriConf/src/dashboard;
    }

A screenshot of dashboard looks like this: pics1 You may click here to see an instance of cloriConf zookeeper dashboard.

API Reference

instance

Config* Config::instance()

Description

Get a Config class instance using singleton pattern

Return value

A Config class pointer which pointer to a Config instance

Load

Config* Load(const std::string& input, uint32_t mode, std::string* err_msg = NULL)

Description

Load configuration data from a string, local file or zookeeper

Parameters

input - config data source, maybe a config string, file name, or zookeeper config file, determined by parameter mode

mode - load mode, consists three parts: source, format and comment(joined by '|'), see the following table for detail

err_msg - optional, filled with detailed error message when loading data error occurred

Mode and their descriptions:

Flag Type Description
SRC_LOCAL source Load config data from local file
SRC_DIRECT source Load config data from input string directly
SRC_ZK source Load config data from zookeeper
FMT_JOML format Parse data as JOML-style/INI-style
FMT_JSON format Parse data as JSON-style
CMT_SHARP comment Take '#' as line comment identifier
CMT_SLASH comment Take '//' as line comment identifier
CMT_SEMICOLON comment Take ';' as line comment identifier
CMT_PERCENT comment Take '%' as line comment identifier

Note

  • Comment type is useful only when format type is FMT_JOML
  • When source type is SRC_ZK, format and comment type are disabled
  • Commment type support combine of multi comment, e.g. "CMT_SHARP|CMT_SLASH" means '#' and '//' are all regard as comment identifier

Example

Config conf;
// Load config data from local file and take ';' as line comment identifier
conf.Load("conf.ini", SRC_LOCAL | FMT_JOML | CMT_SEMICOLON);
// Load config data from zookeeper
conf.Load("zk.ini", SRC_ZK);
// Load JSON-style string config data
std::string input_str("{"myself":{"name":"WeiJian", "school":"BUAA","dr":["cloris", "apache"]}}");
conf.Load(input_str, SRC_DIRECT | FMT_JSON);

Return value

A pointer to config object, if loading failed, a NULL pointer is returned

LoadEx

bool LoadEx(const std::string& input, uint32_t mode, std::string* err_msg = NULL)

Description

Have the same function as Load, the difference is that return value tyle is bool.

Return value

A bool value indicating success or failure.

Example

Config conf;
// Load config data from local file and take ';' as line comment identifier
bool ok = conf.LoadEx("conf.ini", SRC_LOCAL | FMT_JOML | CMT_SEMICOLON);
// Load config data from zookeeper
bool ok = conf.LoadEx("zk.ini", SRC_ZK);
// Load JSON-style string config data
std::string input_str("{"myself":{"name":"WeiJian", "school":"BUAA","dr":["cloris", "apache"]}}");
bool ok = conf.LoadEx(input_str, SRC_DIRECT | FMT_JSON);

GetConfNode

ConfNode* Config::GetConfNode(const std::string& name = "") const
ConfNode* ConfNode::GetConfNode(const std::string& name = "") const

Description

Get specific ConfNode from configuration tree

Parameters

name - path of config node. CloriConf supports two kinds of path style:"p1.p2.p3" or "p1/p2/p3"

GetString/GetInt32/GetInt64/GetDouble/GetBool

std::string Config::GetString(const std::string& name, const std::string& default_value = "") const
int32_t Config::GetInt32(const std::string& name, int32_t default_value = 0) const
int64_t Config::GetInt64(const std::string& name, int64_t default_value = 0L) const
double Config::GetDouble(const std::string& name, double default_value = 0.0) const
bool Config::GetBool(const std::string& name, bool default_value = false) const

std::string ConfNode::GetString(const std::string& name, const std::string& default_value = "") const
int32_t ConfNode::GetInt32(const std::string& name, int32_t default_value = 0) const
int64_t ConfNode::GetInt64(const std::string& name, int64_t default_value = 0L) const
double ConfNode::GetDouble(const std::string& name, double default_value = 0.0) const
bool ConfNode::GetBool(const std::string& name, bool default_value = false) const

Description

Search children config node from current ConfNode and parse children config node as string/int32/int64/double/bool

Parameters

name - Child config node path

default_value - Default return value when ConfNode not found

AsString/AsInt32/AsInt64/AsDouble/AsBool

const std::string& ConfNode::AsString() const
int32_t ConfNode::AsInt32() const
int64_t ConfNode::AsInt64() const
double ConfNode::AsDouble() const
bool ConfNode::AsBool() const

Description

Parse ConfNode as string/int32/int64/double/bool

Exists

bool Config::Exists(const std::string& name) const
bool ConfNode::Exists(const std::string& name) const

Description

Check if conf node exists

Parameters

name - Config node path

name

const std::string& ConfNode::name()

Description

Name of current conf node

begin

ConstChildrenIterator ConfNode::begin() const
ChildrenIterator begin()

Description

Begin of children config node iterator

end

ConstChildrenIterator ConfNode::end() const
ChildrenIterator ConfNode::end()

Description

End of children config node iterator

Example

// use begin and end to traverse children config nodes 
ConfNode* node = conf->GetConfNode("rules/splash");
if (node) { 
    for (ConfNode::ChildrenIterator iter = node->begin(); iter != node->end(); ++iter) {
        std::cout << "zookeeper config node, key=" << iter->name() << ", value=" << iter->AsString() << std::endl;
    }
}

About JOML

JOML(James's Obvious Minimal Language) is a self-defined configuration description language inspired by ini and toml. An instance of joml:

# @ad_server config
[[[ad_server]]]
    [[inmobi]]
        # online config
        host=api.w.xxx.cn
        port=80
        uri=/showadxxxx/v3.1
        # test
        #host=192.168.19.229
        #port=7761
        #uri=/getAdTestIxxx
        timeout_connect=200
        timeout_receive=300 
        retry=0
        [os_config.ios]
            appid=1231
        [os_config.android]
            appid=1231
        [splash.android]
            id=1510609xxx
            bundle=so.xxx.xxx
        [splash.ios]
            id=15119685xxx6
            bundle=com.xxx.xxx

You can treat JOML as a superset of INI, compared with INI, JOML has the following features

  • Section nesting support - e.g. "[[...]]" is the parent section of "[...]", the more brackets nested, the higher layer it is
  • More slack writing style - for example, to express item "ad_server.inmobi.os_config.ios.appid=1231", the following styles are all OK
# case 1
[[[[ad_server]]]]
    [[[inmobi]]]
        [[os_config]]
            [ios]
                appid=1231
# case 2
[ad_server.inmobi.os_config]
    ios.appid=1231
# case 3
[[[ad_server.inmobi]]]
    [os_config]
        ios.appid=1231
  • More line comments identifier are supported - you can specify your own identifier in Load/LoadEx function

Array Access

As designed, cloriConf's main purpose is to support config style like joml and zookeeper which do not have concept of array. To support config style which has array definition(e.g. json,toml), cloriConf translate any array into name=value pair, where name counts from 0 until the end of the array.

The following example shows how to access arry in JSON-style configuration file:

# conf.json:
{
    "id":"N1",
    "name":"shpilu",
    "info":{
        "month":["January","February", "March", "April", "May"]
    }
}

C++ code:

    std::string err_msg;
    Config* conf = Config::instance()->Load("conf.json", SRC_LOCAL | FMT_JSON, &err_msg);
    if (!conf) {
        std::cout << "run test failed, " << err_msg << std::endl;
        return;
    }   
    std::string third_month = conf->GetString("info.month.2");
    // output: the third month is March
    std::cout << "the third month is " << third_month << std::endl;
    ConfNode* node = conf->GetConfNode("info.month");
    if (node) {
        for (auto &p : *node) {
            // 0, January
            // 1, February
            // ...
            std::cout << "name=" << p.name() << ", value=" << p.AsString() << std::endl;
        }   
    }

Who Is Using CloriConf?

  • ofo 小黄车 - ofo Inc., a Beijing-based bicycle sharing company

Related Projects

To some extent, cloriConf is inspired by the upper projects

TODO List

  • Support YAML-style configuration file
  • Support TOML-style configuration file
  • Support gflags-style configuration file
  • Support protobuf-style configuration file
  • Support loading config from ectd
  • Update config automatically and smoothly when config in zookeeper/ectd changed

Authors

Go back to top