mirror of
https://github.com/XRPLF/rippled.git
synced 2025-11-19 18:45:52 +00:00
* Some runner member functions are now thread-safe. * De-inline and tidy up declarations and definitions. * arg() interface allows command lines to be passed to suites.
347 lines
7.5 KiB
C++
347 lines
7.5 KiB
C++
//------------------------------------------------------------------------------
|
|
/*
|
|
This file is part of Beast: https://github.com/vinniefalco/Beast
|
|
Copyright 2013, Vinnie Falco <vinnie.falco@gmail.com>
|
|
|
|
Permission to use, copy, modify, and/or distribute this software for any
|
|
purpose with or without fee is hereby granted, provided that the above
|
|
copyright notice and this permission notice appear in all copies.
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
|
WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
|
MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
|
ANY SPECIAL , DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
|
WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
|
ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
|
OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
*/
|
|
//==============================================================================
|
|
|
|
#ifndef BEAST_UNIT_TEST_RUNNER_H_INCLUDED
|
|
#define BEAST_UNIT_TEST_RUNNER_H_INCLUDED
|
|
|
|
#include <beast/unit_test/suite_info.h>
|
|
#include <beast/streams/abstract_ostream.h>
|
|
#include <cassert>
|
|
#include <mutex>
|
|
#include <string>
|
|
|
|
namespace beast {
|
|
namespace unit_test {
|
|
|
|
/** Unit test runner interface.
|
|
Derived classes can customize the reporting behavior. This interface is
|
|
injected into the unit_test class to receive the results of the tests.
|
|
*/
|
|
class runner
|
|
{
|
|
private:
|
|
// Reroutes log output to the runner
|
|
class stream_t : public abstract_ostream
|
|
{
|
|
private:
|
|
runner& owner_;
|
|
|
|
public:
|
|
stream_t() = delete;
|
|
stream_t& operator= (stream_t const&) = delete;
|
|
|
|
template <class = void>
|
|
stream_t (runner& owner);
|
|
|
|
void
|
|
write (string_type const& s) override
|
|
{
|
|
owner_.log (s);
|
|
}
|
|
};
|
|
|
|
stream_t stream_;
|
|
std::string arg_;
|
|
bool default_ = false;
|
|
bool failed_ = false;
|
|
bool cond_ = false;
|
|
std::recursive_mutex mutex_;
|
|
|
|
public:
|
|
virtual ~runner() = default;
|
|
runner (runner const&) = default;
|
|
runner& operator= (runner const&) = default;
|
|
|
|
template <class = void>
|
|
runner();
|
|
|
|
/** Set the argument string.
|
|
The argument string is available to suites and
|
|
allows for customization of the test. Each suite
|
|
defines its own syntax for the argumnet string.
|
|
The same argument is passed to all suites.
|
|
*/
|
|
void
|
|
arg (std::string const& s)
|
|
{
|
|
arg_ = s;
|
|
}
|
|
|
|
/** Returns the argument string. */
|
|
std::string const&
|
|
arg() const
|
|
{
|
|
return arg_;
|
|
}
|
|
|
|
/** Run the specified suite.
|
|
@return `true` if any conditions failed.
|
|
*/
|
|
template <class = void>
|
|
bool
|
|
run (suite_info const& s);
|
|
|
|
/** Run a sequence of suites.
|
|
The expression
|
|
`FwdIter::value_type`
|
|
must be convertible to `suite_info`.
|
|
@return `true` if any conditions failed.
|
|
*/
|
|
template <class FwdIter>
|
|
bool
|
|
run (FwdIter first, FwdIter last);
|
|
|
|
/** Conditionally run a sequence of suites.
|
|
pred will be called as:
|
|
@code
|
|
bool pred (suite_info const&);
|
|
@endcode
|
|
@return `true` if any conditions failed.
|
|
*/
|
|
template <class FwdIter, class Pred>
|
|
bool
|
|
run_if (FwdIter first, FwdIter last, Pred pred = Pred{});
|
|
|
|
/** Run all suites in a container.
|
|
@return `true` if any conditions failed.
|
|
*/
|
|
template <class SequenceContainer>
|
|
bool
|
|
run_each (SequenceContainer const& c);
|
|
|
|
/** Conditionally run suites in a container.
|
|
pred will be called as:
|
|
@code
|
|
bool pred (suite_info const&);
|
|
@endcode
|
|
@return `true` if any conditions failed.
|
|
*/
|
|
template <class SequenceContainer, class Pred>
|
|
bool
|
|
run_each_if (SequenceContainer const& c, Pred pred = Pred{});
|
|
|
|
private:
|
|
//
|
|
// Overrides
|
|
//
|
|
|
|
/** Called when a new suite starts. */
|
|
virtual
|
|
void
|
|
on_suite_begin (suite_info const&)
|
|
{
|
|
}
|
|
|
|
/** Called when a suite ends. */
|
|
virtual
|
|
void
|
|
on_suite_end()
|
|
{
|
|
}
|
|
|
|
/** Called when a new case starts. */
|
|
virtual
|
|
void
|
|
on_case_begin (std::string const&)
|
|
{
|
|
}
|
|
|
|
/** Called when a new case ends. */
|
|
virtual
|
|
void
|
|
on_case_end()
|
|
{
|
|
}
|
|
|
|
/** Called for each passing condition. */
|
|
virtual
|
|
void
|
|
on_pass ()
|
|
{
|
|
}
|
|
|
|
/** Called for each failing condition. */
|
|
virtual
|
|
void
|
|
on_fail (std::string const&)
|
|
{
|
|
}
|
|
|
|
/** Called when a test logs output. */
|
|
virtual
|
|
void
|
|
on_log (std::string const&)
|
|
{
|
|
}
|
|
|
|
private:
|
|
friend class suite;
|
|
|
|
abstract_ostream&
|
|
stream()
|
|
{
|
|
return stream_;
|
|
}
|
|
|
|
// Start a new testcase.
|
|
template <class = void>
|
|
void
|
|
testcase (std::string const& name);
|
|
|
|
template <class = void>
|
|
void
|
|
pass();
|
|
|
|
template <class = void>
|
|
void
|
|
fail (std::string const& reason);
|
|
|
|
template <class = void>
|
|
void
|
|
log (std::string const& s);
|
|
};
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
template <class>
|
|
runner::stream_t::stream_t (runner& owner)
|
|
: owner_ (owner)
|
|
{
|
|
}
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
template <class>
|
|
runner::runner()
|
|
: stream_ (*this)
|
|
{
|
|
}
|
|
|
|
template <class>
|
|
bool
|
|
runner::run (suite_info const& s)
|
|
{
|
|
// Enable 'default' testcase
|
|
default_ = true;
|
|
failed_ = false;
|
|
on_suite_begin (s);
|
|
s.run (*this);
|
|
// Forgot to call pass or fail.
|
|
assert (cond_);
|
|
on_case_end();
|
|
on_suite_end();
|
|
return failed_;
|
|
}
|
|
|
|
template <class FwdIter>
|
|
bool
|
|
runner::run (FwdIter first, FwdIter last)
|
|
{
|
|
bool failed (false);
|
|
for (;first != last; ++first)
|
|
failed = run (*first) || failed;
|
|
return failed;
|
|
}
|
|
|
|
template <class FwdIter, class Pred>
|
|
bool
|
|
runner::run_if (FwdIter first, FwdIter last, Pred pred)
|
|
{
|
|
bool failed (false);
|
|
for (;first != last; ++first)
|
|
if (pred (*first))
|
|
failed = run (*first) || failed;
|
|
return failed;
|
|
}
|
|
|
|
template <class SequenceContainer>
|
|
bool
|
|
runner::run_each (SequenceContainer const& c)
|
|
{
|
|
bool failed (false);
|
|
for (auto const& s : c)
|
|
failed = run (s) || failed;
|
|
return failed;
|
|
}
|
|
|
|
template <class SequenceContainer, class Pred>
|
|
bool
|
|
runner::run_each_if (SequenceContainer const& c, Pred pred)
|
|
{
|
|
bool failed (false);
|
|
for (auto const& s : c)
|
|
if (pred (s))
|
|
failed = run (s) || failed;
|
|
return failed;
|
|
}
|
|
|
|
template <class>
|
|
void
|
|
runner::testcase (std::string const& name)
|
|
{
|
|
std::lock_guard<std::recursive_mutex> lock(mutex_);
|
|
// Name may not be empty
|
|
assert (default_ || ! name.empty());
|
|
// Forgot to call pass or fail
|
|
assert (default_ || cond_);
|
|
if (! default_)
|
|
on_case_end();
|
|
default_ = false;
|
|
cond_ = false;
|
|
on_case_begin (name);
|
|
}
|
|
|
|
template <class>
|
|
void
|
|
runner::pass()
|
|
{
|
|
std::lock_guard<std::recursive_mutex> lock(mutex_);
|
|
if (default_)
|
|
testcase ("");
|
|
on_pass();
|
|
cond_ = true;
|
|
}
|
|
|
|
template <class>
|
|
void
|
|
runner::fail (std::string const& reason)
|
|
{
|
|
std::lock_guard<std::recursive_mutex> lock(mutex_);
|
|
if (default_)
|
|
testcase ("");
|
|
on_fail (reason);
|
|
failed_ = true;
|
|
cond_ = true;
|
|
}
|
|
|
|
template <class>
|
|
void
|
|
runner::log (std::string const& s)
|
|
{
|
|
std::lock_guard<std::recursive_mutex> lock(mutex_);
|
|
if (default_)
|
|
testcase ("");
|
|
on_log (s);
|
|
}
|
|
|
|
} // unit_test
|
|
} // beast
|
|
|
|
#endif
|