The library is designed to provide a uniform interface for three distributed KV storages: etcd, ZooKeeper and Consul.
The services have similar but different data models, so we outlined the common features.
In our implementation, keys form a ZK-like hierarchy. Each key has a version that is int64 number greater than 0. Current version is returned with other data by the most of operations. All the operations supported are listed below.
Method | Parameters | Description |
---|---|---|
create |
key: string value: char[] leased: bool (=false) -- makes the key to be deleted on client disconnect |
Creates the key. Throws an exception if the key already exists or preceding entry does not exist. Returns: version of the newly created key. |
set | key: string value: char[] |
Assigns the value. Creates the key if it doesn’t exist. Throws an exception if preceding entry does not exist. Returns: new version of the key. |
cas | key: string value: char[] version: int64 (=0) -- expected version of the key |
Compare and set operation. If the key does not exist and the version passed equals 0, creates it. Throws an exception if preceding entry does not exist. If the key exists and its version equals to specified one updates the value. Otherwise does nothing and returns 0. Returns: new version of the key or 0. |
get | key: string watch: bool (=false) -- start watching for change in value |
Returns the value currently assigned to the key. Throws an exception if the key does not exist. If watch is true, creates WatchHandler waiting for a change in value. (see usage example below). Returns: current value and WatchHandler. |
exists | key: string watch: bool (=false) -- start watching for removal or creation of the key |
Checks if the key exists. If watch is true, creates WatchHandler waiting for a change in state of existance (see usage example below). Returns: version of the key or 0 if it doesn't exist and WatchHandler. |
get_children | key: string watch: bool (=false) |
Returns a list of the key's direct children. Throws an exception if the key does not exist. If watch is true, creates WatchHandler waiting for any changes among the children. Returns: list of direct children and WatchHandler. |
erase | key: string version: int64 (=0) |
Erases the key and all its descendants if the version given equals to the current key's version. Does it unconditionally if version is 0. Throws an exception if the key does not exist. Returns: (void) |
commit | transaction: Transaction | Commits transaction (see transactions API below). If it was failed, throws TxnFailed with an index of the failed operation. Returns: list of new versions of keys affected by the transaction |
Transaction is a chain of operations of 4 types: create, set, erase, check, performing atomically. Their descriptions can be found below. N.b. at the moment set has different behavior in comparison to ordinary set: when used in transaction, it does not create the key if it does not exist. Besides you cannot assign watches. Leases are still available. Transaction body is separated into two blocks: firstly you should write all required checks and then the sequence of other operations (see an example below). As a result a list of new versions for all the keys involved in set operations is returned.
Method | Parameters | Description |
---|---|---|
create |
key: string value: char[] leased: bool (=false) |
Creates the key. Rolls back if the key already exists or preceding entry does not exist. |
set | key: string value: char[] |
Set the value. Rolls back if the key does not exist. n.b behaviors of transaction set and ordinary set differ |
erase | key: string version: int64 (=0) |
Erases the key and all its descendants if the version passed equals to the current key's version. Does it unconditionally if the version is 0. Rolls back if the key does not exist. |
check | key: string version: int64 |
Checks if the given key has the specified version. Only checks if it exists if the version is 0 |
#include <iostream>
#include <thread>
#include <liboffkv/liboffkv.hpp>
using namespace liboffkv;
int main()
{
// firstly specify protocol (zk | consul | etcd) and address
// you can also specify a prefix all the keys will start with
auto client = open("consul://127.0.0.1:8500", "/prefix");
// on failure methods throw exceptions (for more details see "liboffkv/client.hpp")
try {
int64_t initial_version = client->create("/key", "value");
std::cout << "Key \"/prefix/key\" was created successfully! "
<< "Its initial version is " << initial_version
<< std::endl;
} catch (EntryExists&) {
// other exception types can be found in liboffkv/errors.hpp
std::cout << "Error: key \"/prefix/key\" already exists!"
<< std::endl;
}
// WATCH EXAMPLE
auto result = client.exists("/key", true);
// this thread erase the key after 10 seconds
std::thread([&client]() mutable {
std::this_thread::sleep_for(std::chrono::seconds(10));
client->erase("/key");
}).detach();
// now the key exists
assert(result);
// wait for changes
result.watch->wait();
// if the waiting was completed, the existance state must be different
assert(!client.exists("/key"));
// TRANSACTION EXAMPLE
// n.b. checks and other ops are separated from each other
try {
auto txn_result = client->commit(
{
// firstly list your checks
{
TxnCheck("/key", 42u),
TxnCheck("/foo"),
},
// then a chain of ops that are to be performed
// in case all checks are satisfied
{
TxnErase("/key"),
TxnSet("/foo", "new_value"),
}
}
);
// only one set/create operation
assert(txn_result.size() == 1 &&
txn_result[0].kind == TxnOpResult::Kind::SET);
std::cout << "After the transaction the new version of \"/foo\" is "
<< txn_result[0].version << std::endl;
} catch (TxnFailed& e) {
// TxnFailed exception contains failed op index
std::cout << "Transaction failed. Failed op index: "
<< e.failed_op() << std::endl;
}
}
The library is currently tested on
-
Ubuntu 18.04
Full support.
-
MacOS
Full support.
-
Windows 10
Only Consul is supported.
-
C++ compiler
Currently tested compilers are
- VS 2019
- g++ 7.4.0
- clang
VS 2017 is known to fail.
-
We suggest using cmake bundled with vcpkg.
- Install dependencies
# from vcpkg root
vcpkg install ppconsul offscale-libetcd-cpp zkpp
Installing all three packages is not required. See control flags at the next step.
- Build tests
# from liboffkv directory
mkdir cmake-build-debug && cd $_
cmake -DCMAKE_BUILD_TYPE=Debug \
-DCMAKE_TOOLCHAIN_FILE="<replace with path to vcpkg.cmake>" \
-DBUILD_TESTS=ON ..
cmake --build .
You can control the set of supported services with the following flags
- `-DENABLE_ZK=[ON|OFF]`
- `-DENABLE_ETCD=[ON|OFF]`
- `-DENABLE_CONSUL=[ON|OFF]`
Sometimes you may also need to specify `VCPKG_TARGET_TRIPLET`.
-
Run tests
# from liboffkv/cmake-build-debug directory make test
We provide a pure C interface. It can be found in liboffkv/clib.h.
Set -DBUILD_CLIB=ON
option to build the library.
- Rust: rsoffkv
- Java: liboffkv-java
- Go: goffkv, goffkv-etcd, goffkv-zk, goffkv-consul
Licensed under any of:
- Apache License, Version 2.0 (LICENSE-APACHE or https://www.apache.org/licenses/LICENSE-2.0)
- MIT license (LICENSE-MIT or https://opensource.org/licenses/MIT)
- CC0 license (LICENSE-CC0 or https://creativecommons.org/publicdomain/zero/1.0/legalcode)
at your option.
Unless you explicitly state otherwise, any contribution intentionally submitted for inclusion in the work by you, as defined in the Apache-2.0 license, shall be licensed as above, without any additional terms or conditions.