A lightweight header-only Clang/GCC compatible C++ library for expressive pattern matching and error handling.
Check an example in examples folder or go to compiler-explorer to play with it!
A header-only C++ library that offers exceptionless error handling and type-safe enums, bringing Rust-inspired error propagation with the ? operator and the match operator to C++.
The expect macro simplifies error propagation by automatically returning the error from a function when an error (Err) is encountered, and it continues with the success value (Ok) otherwise. This results in code that is both safe and easy to read without unnecessary verbosity.
It allows to defer the error-checking to the point where it is really important while ensuring that all the error options are handled.
While I have not performed intensive testing, and performance can greately depend on the compiler version, I did perform some benchs based on: exceptions vs errors from CedarDB
As you can see, cppmatch implementation is equivalent to using std::expected, exceptions seem to be faster in recursive workloads, but I've seen that if you use "empty" types, cppmatch/expected code generation is better.
Results gotten from WSL / i9 9900K
Run on (16 X 3600.01 MHz CPU s)
CPU Caches:
L1 Data 32 KiB (x8)
L1 Instruction 32 KiB (x8)
L2 Unified 256 KiB (x8)
L3 Unified 16384 KiB (x1)
Load Average: 0.43, 0.35, 0.39
---------------------------------------------------------------------------------
Benchmark Time CPU Iterations
---------------------------------------------------------------------------------
recursive_fib_std_expected 3462 ns 3462 ns 197520
recursive_fib_cppmatch 3487 ns 3446 ns 199569
recursive_fib_cppmatch_with_exceptions 5455 ns 5455 ns 135089
recursive_fib_throws 1661 ns 1661 ns 441885
coord_expected 989 ns 989 ns 703142
coord_cppmatch 987 ns 987 ns 580757
coord_cppmatch_with_exceptions 1132 ns 1132 ns 569379
coord_throws 1220 ns 1220 ns 589315
to run the tests perform nix command:
nix run github.com:rucadi/cppmatch#benchmark
A type alias for std::variant<T, E>
, representing either a success value of type T
(index 0) or an error value of type E
(index 1).
- Example:
cppmatch::Result<int, std::string> r = 42; // Success value cppmatch::Result<int, std::string> e = std::string("fail"); // Error value
A type alias for std::variant<Ts...>
, allowing multiple error types to be represented in a single variant.
- Example:
cppmatch::Error<std::string, int> err = 404; // Error can be string or int
A macro that evaluates expr
, which must be a Result<T, E>
. If the result holds a success value (T
), it returns that value. If it holds an error (E
), it returns the error from the function propagating it up the call stack, essentially early-returning in case of error. This is useful for early returns in functions that return a Result
.
- Requirements: The function using
expect
must return aResult
type to handle the error case properly. - Example:
cppmatch::Result<double, std::string> process(std::string_view floating_number) { double r = expect(parse_to_double(floating_number)); // this line is only executed if parse_to_double succeded, if it failed // the function would early-return, as it happens with operator ? in rust. //we can use the result r as if no error could be returned. for(int i = 2; i < 10; ++i) r *= i; return r; }
This macro is dependent on Statements and Declarations in Expressions or also called braced-groups within expressions.
This extension, supported at least in GCC and Clang compilers, and is used in Linux Kernel linux kernel coding style guide.
MSVC is not supported by this macro, if you want to use it in windows, use MinGW or Clang in visual studio
Applies pattern matching on a variant v
using the provided lambdas. Each lambda handles one of the possible types in the variant. For nested variants, it recursively visits until a non-variant value is reached and applies the appropriate lambda.
Any callable which implements the operator() for the types is valid, so you can create structs overloading the operator () and pass them to this function too.
- Parameters:
v
: The variant to match on.lambdas...
: Variadic lambdas, one for each possible type in the variant.
- Return: The result of the matched lambda.
- Example:
struct Err1{}; struct Err2{}; cppmatch::Result<int, cppmatch::Error<Err1, Err2>> v = 42; auto result = cppmatch::match(v, [](int i) { return "Integer: " + std::to_string(i); }, [](Err1 err) { return "Error1"; }, [](Err2 err) { return "Error2"; } ); // result is "Integer: 42"
Returns the success value of a Result<T, E>
if it is a success, otherwise returns the provided default_value
of type T
.
- Parameters:
result
: AResult<T, E>
.default_value
: A value of typeT
to return ifresult
is an error.
- Return: The success value or the default value.
- Example:
cppmatch::Result<int, std::string> r = std::string("error"); int value = cppmatch::default_expect(r, 0); // value is 0
Transforms the error type of a Result<T, E1>
into a Result<T, E2>
by applying the function f
to the error if present. If the Result
is a success, it remains unchanged.
- Parameters:
result
: AResult<T, E1>
.f
: A function that mapsE1
toE2
.
- Return: A
Result<T, E2>
. - Example:
struct Error1{ std::string_view message;}; struct Error2{ std::string_view err;} cppmatch::Result<int, Error1> r = Error1{"hello world"}; auto mapped = cppmatch::map_error(r, [](const Error1& s) { return Error2{s.message}; }); // mapped is Result<int, Error2> with error "hello world"
A range adaptor that takes a range of Result<T, E>
and returns a view containing only the success values (T
). It filters out errors and extracts the success values using std::get<0>
.
- Usage: Can be used with the pipe operator (
|
) or as a function call. - Example:
std::vector<cppmatch::Result<int, std::string>> results = { 1, std::string("error"), 2 }; auto success_values = results | cppmatch::successes; // success_values is a view containing 1 and 2
When exceptions are enabled cppmatch also provides exception-based alternatives to handle error propagation and pattern matching, which supports MSVC
The function expect_e
is an alternative to the macro expect
. It evaluates an expression that returns a Result<T, E>
. If the result is an error, it uses pattern matching to throw the contained error as an exception. Otherwise, it returns the success value.
The try...catch will be generated dynamically with the macro match_e, surrounding the expression between try/catch is not necessary
- Example:
cppmatch::Result<int, Error<std::string>> parse_int(std::string_view input) { // Returns either a valid int or an error string. } int value = expect_e(parse_int("123"));
The match_e
macro allows you to perform pattern matching on the result of a function call that returns a Result
. Unlike match
, it supports exceptions: if the function call throws an exception (via expect_e
or otherwise), match_e
catches it and applies the provided lambdas to handle the exception.
-
Usage Details:
- EXPR: Must be a function call expression (not an lvalue), as enforced by a static assertion.
- Lambdas: A set of lambdas to match the success or error cases.
- Note: This macro is only available when exceptions are enabled.
-
Example:
std::string result_message = match_e(parse_int("abc"), [](int value) { return "Parsed value: " + std::to_string(value); }, [](auto&&) { return "An error has occured." } );
Under the hood, match_e
wraps the function call in a lambda, executes it, and if an exception is thrown, it attempts to match the exception against the provided lambdas by iterating through the possible error types. If no match is found, it rethrows the exception.