A type-safe, event-driven, thread-safe C++ configuration system with TOML persistence.
It treats configuration as strongly typed variables, supports rich STL and chrono types, provides change events, and allows full customization via Codec<T>.
- Strongly typed fields
Field<T>/FieldValue<T>behave like normal variables- Compile-time constraint:
Serializable
- Automatic TOML mapping
- Hierarchical paths via
::(e.g.net::http::port) - Automatically builds nested TOML tables
- Hierarchical paths via
- Event-driven
VALUE_LOADwhen loaded from fileVALUE_CHANGwhen modified at runtime
- Thread-safe
- Separate locks for value / TOML / events
- Extensible serialization
- Built-in support for many STL, chrono, and utility types
- User-defined types via
Codec<T>
- Batch save
- Changes are marked dirty
Config::save()persists all modified fields at once
- C++20
- ToruNiina/toml11 ≥ 4.4.0
- Neargye/magic_enum ≥ 0.9.7
- stdpp-event
#include "config.h"
using namespace stdpp::config;
Field<int> a("x");
Field<int> b("x", 255);
Field<int> c("x::y", 5);- Fields are uniquely identified by their full path string
- Fields with the same name and type share the same storage
- The first constructed field decides the default value
- Later declarations with the same name:
- Same type → reuse existing value, ignore default
- Different type → throws exception
| Declaration | Path | Shared | Default Used |
|---|---|---|---|
Field<int> a("x") |
x |
yes | int{} |
Field<int> b("x", 255) |
x |
yes | ignored |
Field<int> c("x::y", 5) |
x::y |
no | 5 |
FieldValue<T> behaves like T if T supports the operator.
=- assign from
T - assign from
FieldValue<T>
+ - * /+= -= *= /=
| & ^|= &= ^=
<< >><<= >>=
++x x++--x x--
operator()for function-like types
i = field + 1i += fieldfield += other_field
All operators are only enabled if the underlying
Tsupports them.
std::vector<T>std::list<T>std::deque<T>std::forward_list<T>std::array<T, N>
std::queue<T>std::stack<T>std::priority_queue<T>
std::set<T>std::multiset<T>std::map<K, V>std::multimap<K, V>std::unordered_map<K, V>
std::pair<T1, T2>std::tuple<Ts...>std::optional<T>std::variant<Ts...>std::expected<T, E>std::complex<T>std::bitset<N>std::filesystem::pathstd::atomic<T>
std::chrono::durationstd::chrono::hh_mm_ssstd::chrono::sys_timestd::chrono::year_month_daystd::chrono::zoned_time
std::unique_ptr<T>std::shared_ptr<T>
- Any
enumorenum class
Serialized as string names viamagic_enum
Field<int> port("server::port", 8080);
Field<int> port2("server::port", 8080); // Ignore 8080 Parameter
// port == port2
Field<std::vector<int>> vec("test::vec", {1,2,3});
Config::load("config.toml");
Field<std::optional<int>> opt("test::opt", std::nullopt);
Field<Test> mode("app::mode", Test::A);
mode.change();
opt = std::nullopt; // opt.change();
Config::save(); // change only
TOML:
[server]
port = 8080
[test]
vec = [1,2,3]
[test.opt]
has = false
# value = 114
[app]
mode = "A"auto h = port.add_event([](auto&, Event ev){
if(ev == Event::VALUE_CHANG) { /* changed */ }
});Event types:
enum class Event {
VALUE_CHANG,
VALUE_LOAD
};Define a Codec<T> specialization:
struct Point { int x; int y; };
template<>
struct Codec<Point> {
static toml::value encode(const Point& p) {
return { {"x", p.x}, {"y", p.y} };
}
static Point decode(const toml::value& v) {
return { v.at("x"), v.at("y") };
}
};Usage:
Field<Point> pos("window::pos", {10,20});Each field internally has:
value_mutex– protects valuetoml_mutex– protects encode/decodeevent_mutex– protects callbacks
Manual lock:
auto lock = field.write_lock();
// modify safely- Global static
Config - Fields cannot be removed at runtime
- Type mismatch on same name throws exception
- File is created only on first successful
save()