A header-only C++ library for reusable utility functions
LBNLCPPCommon is a header-only C++ library designed to provide commonly used functions and utilities across multiple projects. It eliminates code duplication by offering a centralized set of generic templates and helper functions.
- Header-only - no compilation needed
- Modern C++20 features - concepts, ranges,
std::optional - Lightweight and easy to integrate with CMake
- No dependencies required beyond the standard library
- Thread-safe where applicable (e.g., LazyEvaluator)
LBNLCPPCommon/
├── include/
│ └── lbnl/
│ ├── algorithm.hxx # Container and range algorithms
│ ├── optional.hxx # OptionalExt with monadic operations
│ ├── optional_utils.hxx # Optional utility functions
│ ├── expected.hxx # ExpectedExt for error handling
│ ├── map_utils.hxx # Associative container utilities
│ ├── enum_index_mapper.hxx # Bidirectional enum-index mapping
│ └── memoize.hxx # LazyEvaluator for caching
├── docs/ # Detailed documentation
├── tst/ # Unit tests
├── CMakeLists.txt
├── LICENSE
└── README.md
The library itself is header-only, so consumers don't compile anything. This section is for developers contributing to LBNLCPPCommon — building the test suite and running it locally.
All dependencies (currently just GoogleTest) are downloaded automatically via CMake FetchContent on first configure. CMake presets are provided for convenience.
CMakePresets.json ships four visible configure presets, plus two hidden inheritance bases used by both the shipped variants and any personal compiler presets:
| Preset | When to use it |
|---|---|
default-debug / default-release |
Standard configure on any platform; CI. Fetches everything from declared remotes. Picks the system default compiler (MSVC on Windows, system cc/c++ on Linux/macOS). |
local-debug / local-release |
Consume sibling working copies of dependencies instead of fetching them. |
Examples:
cmake --preset default-release
cmake --preset local-debug
local expects a sibling directory layout — ../googletest next to ../LBNLCPPCommon. Currently overridden:
| Dependency | Expected path |
|---|---|
| googletest | ../googletest |
Missing siblings fall back to the declared remote automatically, so local-* is safe to invoke even if you don't have the sibling checked out.
CMakePresets.json ships only the four default-* / local-* framework presets — no compiler choices. To get explicit per-compiler presets (vs2022-debug, gcc-13-release, clang-18-debug, etc.), each developer maintains their own CMakeUserPresets.json next to CMakePresets.json. It is gitignored, read automatically by CMake (and CLion, VS Code, etc.), and stays on the developer's machine.
Personal presets inherit from one of the shipped presets (usually local, which gives you sibling-repo overrides for free) and override whatever they want. A complete realistic example — building with WSL Clang on a Windows machine, with CLion 2023.2+ routed through the WSL toolchain automatically:
{
"version": 6,
"configurePresets": [
{
"name": "clang-release",
"displayName": "clang (Release)",
"inherits": "local",
"generator": "Ninja",
"binaryDir": "${sourceDir}/build/clang-release",
"cacheVariables": {
"CMAKE_C_COMPILER": "clang",
"CMAKE_CXX_COMPILER": "clang++",
"CMAKE_BUILD_TYPE": "Release"
},
"vendor": {
"jetbrains.com/clion": {
"toolchain": "WSL"
}
}
}
]
}A few things going on in that one preset:
"inherits": "local"→ picks up sibling-repo overrides (when present) and the rest of the framework setup.- Bare compiler names (
clang,clang++) rather than/usr/bin/clang→ portable to any machine that has that toolchain onPATH. Use absolute paths only if the compiler isn't onPATH(e.g.C:/Program Files/LLVM/bin/clang.exe— forward slashes work in JSON, no escaping needed). "vendor.jetbrains.com/clion.toolchain"→ tells CLion (2023.2+) which configured toolchain to route this preset through. Standard names areWSL,Visual Studio,MinGW; whatever you see inSettings → Build, Execution, Deployment → Toolchains. The hint is silently ignored if the name doesn't match — no configure-time error.
Add as many of those blocks as you have toolchains you want explicit presets for (one per compiler × build type). Each gets its own binaryDir so Debug and Release artifacts don't clobber each other.
Alternative if you don't want a personal preset at all: set CC and CXX environment variables in your shell rc (~/.bashrc, PowerShell profile) before invoking cmake --preset default-release. CMake picks them up. Tradeoff — affects every CMake project on your machine, not just this one.
Each preset lands in its own subdirectory under build/:
cmake --build build/default-release --parallel
ctest --test-dir build/default-release --output-on-failure
cmake -B build
cmake --build build --config Release --parallel
ctest --test-dir build --output-on-failure
Delete the build/ directory and re-run the configure and build commands above.
include(FetchContent)
FetchContent_Declare(
LBNLCPPCommon
GIT_REPOSITORY https://github.com/LBNL-ETA/LBNLCPPCommon.git
GIT_TAG main
)
FetchContent_MakeAvailable(LBNLCPPCommon)
target_link_libraries(MyProject PRIVATE LBNLCPPCommon)git submodule add https://github.com/LBNL-ETA/LBNLCPPCommon.git external/LBNLCPPCommon
git submodule update --init --recursiveThen in your CMakeLists.txt:
add_subdirectory(external/LBNLCPPCommon)
target_link_libraries(MyProject PRIVATE LBNLCPPCommon)#include <lbnl/algorithm.hxx>
#include <vector>
std::vector<int> numbers = {1, 2, 3, 4, 5};
auto found = lbnl::find_element(numbers, [](int x) { return x > 3; });
if (found) {
std::cout << "Found: " << *found << '\n'; // Output: Found: 4
}#include <lbnl/optional.hxx>
std::optional<std::string> input = "42";
int result = lbnl::extend(input)
.and_then([](const std::string& s) { return parse_int(s); }) // returns optional<int>
.and_then([](int x) { return x * 2; })
.or_else([]() { return 0; }) // fallback if any step fails
.value_or(0);#include <lbnl/expected.hxx>
using Result = lbnl::ExpectedExt<int, std::string>;
Result safe_divide(int num, int den) {
if (den == 0) return lbnl::Unexpected(std::string("Division by zero"));
return num / den;
}
auto result = safe_divide(10, 2)
.and_then([](int x) { return safe_divide(x, 2); })
.transform([](int x) { return x * 10; });
if (result.has_value()) {
std::cout << result.value() << '\n'; // 25
}#include <lbnl/memoize.hxx>
lbnl::LazyEvaluator<std::string, std::string> fileCache(
[](const std::string& path) {
return read_file_content(path); // expensive operation
}
);
const auto & content1 = fileCache("config.txt"); // reads file
const auto & content2 = fileCache("config.txt"); // returns cached referenceAlgorithm Functions (docs/algorithm.md)
Generic algorithms for containers and ranges.
| Function | Description |
|---|---|
find_element |
Find first element matching a predicate |
contains |
Check if container contains a value |
sorted_unique |
Remove duplicates from a range (sorts first) |
zip |
Combine two ranges into pairs |
filter |
Filter elements by predicate |
transform_if |
Transform matching elements, copy others |
transform_filter |
Transform and filter in one pass |
merge |
Merge two sorted ranges |
split |
Split string by delimiter |
partition |
Split range into two groups |
flatten |
Flatten nested vectors |
to_vector |
Convert any range to vector |
transform_to_vector |
Transform range to vector |
OptionalExt (docs/optional.md)
Extended optional with C++23-like monadic operations.
| Method | Description |
|---|---|
and_then |
Chain optional-returning operations |
or_else |
Provide fallback for empty optional |
map / transform |
Transform contained value |
value_or |
Get value or fallback |
Also includes average_optional() for computing averages of optional vectors.
ExpectedExt (docs/expected.md)
Result type for error handling without exceptions.
| Method | Description |
|---|---|
has_value |
Check if result is success |
value |
Get success value |
error |
Get error value |
value_or |
Get value or fallback |
and_then |
Chain on success (func returns ExpectedExt) |
or_else |
Handle errors |
transform |
Transform success value (func returns plain value) |
transform_error |
Transform error value |
Map Utilities (docs/map_utils.md)
Utilities for associative containers (std::map, std::unordered_map).
| Function | Description |
|---|---|
map_lookup_by_key |
Safe key lookup returning optional |
map_lookup_by_value |
Reverse lookup by value |
map_keys |
Extract all keys as vector |
map_values |
Extract all values as vector |
EnumIndexMapper (docs/enum_index_mapper.md)
Bidirectional mapping between enum values and database indices.
| Method | Description |
|---|---|
to_enum |
Convert index to enum |
to_index |
Convert enum to index |
to_enum_or |
Convert with fallback enum |
to_index_or |
Convert with fallback index |
LazyEvaluator (docs/memoize.md)
Thread-safe memoization for expensive computations.
| Method | Description |
|---|---|
operator() |
Get or compute value (returns const Value &) |
get |
Get or compute value (returns const Value &) |
Detailed documentation for each component is available in the docs/ folder:
- C++20 compatible compiler
- CMake 3.14+
This project is licensed under the Berkeley Lab License - see the LICENSE file for details.
We welcome contributions! Feel free to open issues, suggest improvements, or submit pull requests.
For any questions or suggestions, please reach out to the LBNL-ETA team.