Wire Sysio Wire Sysion 1.0.0
|
Matchers are an alternative way to do assertions which are easily extensible and composable. This makes them well suited to use with more complex types (such as collections) or your own custom types. Matchers were first popularised by the Hamcrest family of frameworks.
Matchers are introduced with the REQUIRE_THAT
or CHECK_THAT
macros, which take two arguments. The first argument is the thing (object or value) under test. The second part is a match expression, which consists of either a single matcher or one or more matchers combined using &&
, ||
or !
operators.
For example, to assert that a string ends with a certain substring:
The matcher objects can take multiple arguments, allowing more fine tuning. The built-in string matchers, for example, take a second argument specifying whether the comparison is case sensitive or not:
And matchers can be combined:
Catch currently provides some matchers, they are in the Catch::Matchers
and Catch
namespaces.
The string matchers are StartsWith
, EndsWith
, Contains
, Equals
and Matches
. The first four match a literal (sub)string against a result, while Matches
takes and matches an ECMAScript regex. Do note that Matches
matches the string as a whole, meaning that "abc" will not match against "abcd", but "abc.*" will.
Each of the provided std::string
matchers also takes an optional second argument, that decides case sensitivity (by-default, they are case sensitive).
The vector matchers are Contains
, VectorContains
and Equals
. VectorContains
looks for a single element in the matched vector, Contains
looks for a set (vector) of elements inside the matched vector.
The floating point matchers are WithinULP
and WithinAbs
. WithinAbs
accepts floating point numbers that are within a certain margin of target. WithinULP
performs an ULP-based comparison of two floating point numbers and accepts them if they are less than certain number of ULPs apart.
Do note that ULP-based checks only make sense when both compared numbers are of the same type and WithinULP
will use type of its argument as the target type. This means that WithinULP(1.f, 1)
will expect to compare float
s, but WithinULP(1., 1)
will expect to compare double
s.
Catch also aims to provide a set of generic matchers. Currently this set contains only a matcher that takes arbitrary callable predicate and applies it onto the provided object.
Because of type inference limitations, the argument type of the predicate has to be provided explicitly. Example:
The second argument is an optional description of the predicate, and is used only during reporting of the result.
It's easy to provide your own matchers to extend Catch or just to work with your own types.
You need to provide two things:
Catch::MatcherBase<T>
- where T
is the type being tested. The constructor takes and stores any arguments needed (e.g. something to compare against) and you must override two methods: match()
and describe()
.Here's an example for asserting that an integer falls within a given range (note that it is all inline for the sake of keeping the example short):
Running this test gives the following in the console: