mirror of
https://github.com/XRPLF/rippled.git
synced 2025-11-19 18:45:52 +00:00
665 lines
15 KiB
C++
665 lines
15 KiB
C++
//
|
|
// Copyright (c) 2013-2017 Vinnie Falco (vinnie dot falco at gmail dot com)
|
|
//
|
|
// Distributed under the Boost Software License, Version 1.0. (See accompanying
|
|
// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
|
|
//
|
|
|
|
#ifndef BEAST_UNIT_TEST_SUITE_HPP
|
|
#define BEAST_UNIT_TEST_SUITE_HPP
|
|
|
|
#include <ripple/beast/unit_test/runner.h>
|
|
#include <boost/filesystem.hpp>
|
|
#include <boost/lexical_cast.hpp>
|
|
#include <boost/throw_exception.hpp>
|
|
#include <ostream>
|
|
#include <sstream>
|
|
#include <string>
|
|
|
|
namespace beast {
|
|
namespace unit_test {
|
|
|
|
namespace detail {
|
|
|
|
template <class String>
|
|
static std::string
|
|
make_reason(String const& reason, char const* file, int line)
|
|
{
|
|
std::string s(reason);
|
|
if (!s.empty())
|
|
s.append(": ");
|
|
namespace fs = boost::filesystem;
|
|
s.append(fs::path{file}.filename().string());
|
|
s.append("(");
|
|
s.append(boost::lexical_cast<std::string>(line));
|
|
s.append(")");
|
|
return s;
|
|
}
|
|
|
|
} // namespace detail
|
|
|
|
class thread;
|
|
|
|
enum abort_t { no_abort_on_fail, abort_on_fail };
|
|
|
|
/** A testsuite class.
|
|
|
|
Derived classes execute a series of testcases, where each testcase is
|
|
a series of pass/fail tests. To provide a unit test using this class,
|
|
derive from it and use the BEAST_DEFINE_UNIT_TEST macro in a
|
|
translation unit.
|
|
*/
|
|
class suite
|
|
{
|
|
private:
|
|
bool abort_ = false;
|
|
bool aborted_ = false;
|
|
runner* runner_ = nullptr;
|
|
|
|
// This exception is thrown internally to stop the current suite
|
|
// in the event of a failure, if the option to stop is set.
|
|
struct abort_exception : public std::exception
|
|
{
|
|
char const*
|
|
what() const noexcept override
|
|
{
|
|
return "test suite aborted";
|
|
}
|
|
};
|
|
|
|
template <class CharT, class Traits, class Allocator>
|
|
class log_buf : public std::basic_stringbuf<CharT, Traits, Allocator>
|
|
{
|
|
suite& suite_;
|
|
|
|
public:
|
|
explicit log_buf(suite& self) : suite_(self)
|
|
{
|
|
}
|
|
|
|
~log_buf()
|
|
{
|
|
sync();
|
|
}
|
|
|
|
int
|
|
sync() override
|
|
{
|
|
auto const& s = this->str();
|
|
if (s.size() > 0)
|
|
suite_.runner_->log(s);
|
|
this->str("");
|
|
return 0;
|
|
}
|
|
};
|
|
|
|
template <
|
|
class CharT,
|
|
class Traits = std::char_traits<CharT>,
|
|
class Allocator = std::allocator<CharT>>
|
|
class log_os : public std::basic_ostream<CharT, Traits>
|
|
{
|
|
log_buf<CharT, Traits, Allocator> buf_;
|
|
|
|
public:
|
|
explicit log_os(suite& self)
|
|
: std::basic_ostream<CharT, Traits>(&buf_), buf_(self)
|
|
{
|
|
}
|
|
};
|
|
|
|
class scoped_testcase;
|
|
|
|
class testcase_t
|
|
{
|
|
suite& suite_;
|
|
std::stringstream ss_;
|
|
|
|
public:
|
|
explicit testcase_t(suite& self) : suite_(self)
|
|
{
|
|
}
|
|
|
|
/** Open a new testcase.
|
|
|
|
A testcase is a series of evaluated test conditions. A test
|
|
suite may have multiple test cases. A test is associated with
|
|
the last opened testcase. When the test first runs, a default
|
|
unnamed case is opened. Tests with only one case may omit the
|
|
call to testcase.
|
|
|
|
@param abort Determines if suite continues running after a failure.
|
|
*/
|
|
void
|
|
operator()(std::string const& name, abort_t abort = no_abort_on_fail);
|
|
|
|
scoped_testcase
|
|
operator()(abort_t abort);
|
|
|
|
template <class T>
|
|
scoped_testcase
|
|
operator<<(T const& t);
|
|
};
|
|
|
|
public:
|
|
/** Logging output stream.
|
|
|
|
Text sent to the log output stream will be forwarded to
|
|
the output stream associated with the runner.
|
|
*/
|
|
log_os<char> log;
|
|
|
|
/** Memberspace for declaring test cases. */
|
|
testcase_t testcase;
|
|
|
|
/** Returns the "current" running suite.
|
|
If no suite is running, nullptr is returned.
|
|
*/
|
|
static suite*
|
|
this_suite()
|
|
{
|
|
return *p_this_suite();
|
|
}
|
|
|
|
suite() : log(*this), testcase(*this)
|
|
{
|
|
}
|
|
|
|
virtual ~suite() = default;
|
|
suite(suite const&) = delete;
|
|
suite&
|
|
operator=(suite const&) = delete;
|
|
|
|
/** Invokes the test using the specified runner.
|
|
|
|
Data members are set up here instead of the constructor as a
|
|
convenience to writing the derived class to avoid repetition of
|
|
forwarded constructor arguments to the base.
|
|
Normally this is called by the framework for you.
|
|
*/
|
|
template <class = void>
|
|
void
|
|
operator()(runner& r);
|
|
|
|
/** Record a successful test condition. */
|
|
template <class = void>
|
|
void
|
|
pass();
|
|
|
|
/** Record a failure.
|
|
|
|
@param reason Optional text added to the output on a failure.
|
|
|
|
@param file The source code file where the test failed.
|
|
|
|
@param line The source code line number where the test failed.
|
|
*/
|
|
/** @{ */
|
|
template <class String>
|
|
void
|
|
fail(String const& reason, char const* file, int line);
|
|
|
|
template <class = void>
|
|
void
|
|
fail(std::string const& reason = "");
|
|
/** @} */
|
|
|
|
/** Evaluate a test condition.
|
|
|
|
This function provides improved logging by incorporating the
|
|
file name and line number into the reported output on failure,
|
|
as well as additional text specified by the caller.
|
|
|
|
@param shouldBeTrue The condition to test. The condition
|
|
is evaluated in a boolean context.
|
|
|
|
@param reason Optional added text to output on a failure.
|
|
|
|
@param file The source code file where the test failed.
|
|
|
|
@param line The source code line number where the test failed.
|
|
|
|
@return `true` if the test condition indicates success.
|
|
*/
|
|
/** @{ */
|
|
template <class Condition>
|
|
bool
|
|
expect(Condition const& shouldBeTrue)
|
|
{
|
|
return expect(shouldBeTrue, "");
|
|
}
|
|
|
|
template <class Condition, class String>
|
|
bool
|
|
expect(Condition const& shouldBeTrue, String const& reason);
|
|
|
|
template <class Condition>
|
|
bool
|
|
expect(Condition const& shouldBeTrue, char const* file, int line)
|
|
{
|
|
return expect(shouldBeTrue, "", file, line);
|
|
}
|
|
|
|
template <class Condition, class String>
|
|
bool
|
|
expect(
|
|
Condition const& shouldBeTrue,
|
|
String const& reason,
|
|
char const* file,
|
|
int line);
|
|
/** @} */
|
|
|
|
//
|
|
// DEPRECATED
|
|
//
|
|
// Expect an exception from f()
|
|
template <class F, class String>
|
|
bool
|
|
except(F&& f, String const& reason);
|
|
template <class F>
|
|
bool
|
|
except(F&& f)
|
|
{
|
|
return except(f, "");
|
|
}
|
|
template <class E, class F, class String>
|
|
bool
|
|
except(F&& f, String const& reason);
|
|
template <class E, class F>
|
|
bool
|
|
except(F&& f)
|
|
{
|
|
return except<E>(f, "");
|
|
}
|
|
template <class F, class String>
|
|
bool
|
|
unexcept(F&& f, String const& reason);
|
|
template <class F>
|
|
bool
|
|
unexcept(F&& f)
|
|
{
|
|
return unexcept(f, "");
|
|
}
|
|
|
|
/** Return the argument associated with the runner. */
|
|
std::string const&
|
|
arg() const
|
|
{
|
|
return runner_->arg();
|
|
}
|
|
|
|
// DEPRECATED
|
|
// @return `true` if the test condition indicates success(a false value)
|
|
template <class Condition, class String>
|
|
bool
|
|
unexpected(Condition shouldBeFalse, String const& reason);
|
|
|
|
template <class Condition>
|
|
bool
|
|
unexpected(Condition shouldBeFalse)
|
|
{
|
|
return unexpected(shouldBeFalse, "");
|
|
}
|
|
|
|
private:
|
|
friend class thread;
|
|
|
|
static suite**
|
|
p_this_suite()
|
|
{
|
|
static suite* pts = nullptr;
|
|
return &pts;
|
|
}
|
|
|
|
/** Runs the suite. */
|
|
virtual void
|
|
run() = 0;
|
|
|
|
void
|
|
propagate_abort();
|
|
|
|
template <class = void>
|
|
void
|
|
run(runner& r);
|
|
};
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
// Helper for streaming testcase names
|
|
class suite::scoped_testcase
|
|
{
|
|
private:
|
|
suite& suite_;
|
|
std::stringstream& ss_;
|
|
|
|
public:
|
|
scoped_testcase&
|
|
operator=(scoped_testcase const&) = delete;
|
|
|
|
~scoped_testcase()
|
|
{
|
|
auto const& name = ss_.str();
|
|
if (!name.empty())
|
|
suite_.runner_->testcase(name);
|
|
}
|
|
|
|
scoped_testcase(suite& self, std::stringstream& ss) : suite_(self), ss_(ss)
|
|
{
|
|
ss_.clear();
|
|
ss_.str({});
|
|
}
|
|
|
|
template <class T>
|
|
scoped_testcase(suite& self, std::stringstream& ss, T const& t)
|
|
: suite_(self), ss_(ss)
|
|
{
|
|
ss_.clear();
|
|
ss_.str({});
|
|
ss_ << t;
|
|
}
|
|
|
|
template <class T>
|
|
scoped_testcase&
|
|
operator<<(T const& t)
|
|
{
|
|
ss_ << t;
|
|
return *this;
|
|
}
|
|
};
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
inline void
|
|
suite::testcase_t::operator()(std::string const& name, abort_t abort)
|
|
{
|
|
suite_.abort_ = abort == abort_on_fail;
|
|
suite_.runner_->testcase(name);
|
|
}
|
|
|
|
inline suite::scoped_testcase
|
|
suite::testcase_t::operator()(abort_t abort)
|
|
{
|
|
suite_.abort_ = abort == abort_on_fail;
|
|
return {suite_, ss_};
|
|
}
|
|
|
|
template <class T>
|
|
inline suite::scoped_testcase
|
|
suite::testcase_t::operator<<(T const& t)
|
|
{
|
|
return {suite_, ss_, t};
|
|
}
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
template <class>
|
|
void
|
|
suite::operator()(runner& r)
|
|
{
|
|
*p_this_suite() = this;
|
|
try
|
|
{
|
|
run(r);
|
|
*p_this_suite() = nullptr;
|
|
}
|
|
catch (...)
|
|
{
|
|
*p_this_suite() = nullptr;
|
|
throw;
|
|
}
|
|
}
|
|
|
|
template <class Condition, class String>
|
|
bool
|
|
suite::expect(Condition const& shouldBeTrue, String const& reason)
|
|
{
|
|
if (shouldBeTrue)
|
|
{
|
|
pass();
|
|
return true;
|
|
}
|
|
fail(reason);
|
|
return false;
|
|
}
|
|
|
|
template <class Condition, class String>
|
|
bool
|
|
suite::expect(
|
|
Condition const& shouldBeTrue,
|
|
String const& reason,
|
|
char const* file,
|
|
int line)
|
|
{
|
|
if (shouldBeTrue)
|
|
{
|
|
pass();
|
|
return true;
|
|
}
|
|
fail(detail::make_reason(reason, file, line));
|
|
return false;
|
|
}
|
|
|
|
// DEPRECATED
|
|
|
|
template <class F, class String>
|
|
bool
|
|
suite::except(F&& f, String const& reason)
|
|
{
|
|
try
|
|
{
|
|
f();
|
|
fail(reason);
|
|
return false;
|
|
}
|
|
catch (...)
|
|
{
|
|
pass();
|
|
}
|
|
return true;
|
|
}
|
|
|
|
template <class E, class F, class String>
|
|
bool
|
|
suite::except(F&& f, String const& reason)
|
|
{
|
|
try
|
|
{
|
|
f();
|
|
fail(reason);
|
|
return false;
|
|
}
|
|
catch (E const&)
|
|
{
|
|
pass();
|
|
}
|
|
return true;
|
|
}
|
|
|
|
template <class F, class String>
|
|
bool
|
|
suite::unexcept(F&& f, String const& reason)
|
|
{
|
|
try
|
|
{
|
|
f();
|
|
pass();
|
|
return true;
|
|
}
|
|
catch (...)
|
|
{
|
|
fail(reason);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
template <class Condition, class String>
|
|
bool
|
|
suite::unexpected(Condition shouldBeFalse, String const& reason)
|
|
{
|
|
bool const b = static_cast<bool>(shouldBeFalse);
|
|
if (!b)
|
|
pass();
|
|
else
|
|
fail(reason);
|
|
return !b;
|
|
}
|
|
|
|
template <class>
|
|
void
|
|
suite::pass()
|
|
{
|
|
propagate_abort();
|
|
runner_->pass();
|
|
}
|
|
|
|
// ::fail
|
|
template <class>
|
|
void
|
|
suite::fail(std::string const& reason)
|
|
{
|
|
propagate_abort();
|
|
runner_->fail(reason);
|
|
if (abort_)
|
|
{
|
|
aborted_ = true;
|
|
BOOST_THROW_EXCEPTION(abort_exception());
|
|
}
|
|
}
|
|
|
|
template <class String>
|
|
void
|
|
suite::fail(String const& reason, char const* file, int line)
|
|
{
|
|
fail(detail::make_reason(reason, file, line));
|
|
}
|
|
|
|
inline void
|
|
suite::propagate_abort()
|
|
{
|
|
if (abort_ && aborted_)
|
|
BOOST_THROW_EXCEPTION(abort_exception());
|
|
}
|
|
|
|
template <class>
|
|
void
|
|
suite::run(runner& r)
|
|
{
|
|
runner_ = &r;
|
|
|
|
try
|
|
{
|
|
run();
|
|
}
|
|
catch (abort_exception const&)
|
|
{
|
|
// ends the suite
|
|
}
|
|
catch (std::exception const& e)
|
|
{
|
|
runner_->fail("unhandled exception: " + std::string(e.what()));
|
|
}
|
|
catch (...)
|
|
{
|
|
runner_->fail("unhandled exception");
|
|
}
|
|
}
|
|
|
|
#ifndef BEAST_EXPECT
|
|
/** Check a precondition.
|
|
|
|
If the condition is false, the file and line number are reported.
|
|
*/
|
|
#define BEAST_EXPECT(cond) expect(cond, __FILE__, __LINE__)
|
|
#endif
|
|
|
|
#ifndef BEAST_EXPECTS
|
|
/** Check a precondition.
|
|
|
|
If the condition is false, the file and line number are reported.
|
|
*/
|
|
#define BEAST_EXPECTS(cond, reason) \
|
|
((cond) ? (pass(), true) : (fail((reason), __FILE__, __LINE__), false))
|
|
#endif
|
|
|
|
} // namespace unit_test
|
|
} // namespace beast
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
// detail:
|
|
// This inserts the suite with the given manual flag
|
|
#define BEAST_DEFINE_TESTSUITE_INSERT( \
|
|
Class, Module, Library, manual, priority) \
|
|
static beast::unit_test::detail::insert_suite<Class##_test> \
|
|
Library##Module##Class##_test_instance( \
|
|
#Class, #Module, #Library, manual, priority)
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
// Preprocessor directives for controlling unit test definitions.
|
|
|
|
// If this is already defined, don't redefine it. This allows
|
|
// programs to provide custom behavior for testsuite definitions
|
|
//
|
|
#ifndef BEAST_DEFINE_TESTSUITE
|
|
|
|
/** Enables insertion of test suites into the global container.
|
|
The default is to insert all test suite definitions into the global
|
|
container. If BEAST_DEFINE_TESTSUITE is user defined, this macro
|
|
has no effect.
|
|
*/
|
|
#ifndef BEAST_NO_UNIT_TEST_INLINE
|
|
#define BEAST_NO_UNIT_TEST_INLINE 0
|
|
#endif
|
|
|
|
/** Define a unit test suite.
|
|
|
|
Class The type representing the class being tested.
|
|
Module Identifies the module.
|
|
Library Identifies the library.
|
|
|
|
The declaration for the class implementing the test should be the same
|
|
as Class ## _test. For example, if Class is aged_ordered_container, the
|
|
test class must be declared as:
|
|
|
|
@code
|
|
|
|
struct aged_ordered_container_test : beast::unit_test::suite
|
|
{
|
|
//...
|
|
};
|
|
|
|
@endcode
|
|
|
|
The macro invocation must appear in the same namespace as the test class.
|
|
|
|
Unit test priorities were introduced so parallel unit_test::suites would
|
|
execute faster. Suites with longer running times have higher priorities
|
|
than unit tests with shorter running times. Suites with no priorities
|
|
are assumed to run most quickly, so they run last.
|
|
*/
|
|
|
|
#if BEAST_NO_UNIT_TEST_INLINE
|
|
#define BEAST_DEFINE_TESTSUITE(Class, Module, Library)
|
|
#define BEAST_DEFINE_TESTSUITE_MANUAL(Class, Module, Library)
|
|
#define BEAST_DEFINE_TESTSUITE_PRIO(Class, Module, Library, Priority)
|
|
#define BEAST_DEFINE_TESTSUITE_MANUAL_PRIO(Class, Module, Library, Priority)
|
|
|
|
#else
|
|
#include <ripple/beast/unit_test/global_suites.h>
|
|
#define BEAST_DEFINE_TESTSUITE(Class, Module, Library) \
|
|
BEAST_DEFINE_TESTSUITE_INSERT(Class, Module, Library, false, 0)
|
|
#define BEAST_DEFINE_TESTSUITE_MANUAL(Class, Module, Library) \
|
|
BEAST_DEFINE_TESTSUITE_INSERT(Class, Module, Library, true, 0)
|
|
#define BEAST_DEFINE_TESTSUITE_PRIO(Class, Module, Library, Priority) \
|
|
BEAST_DEFINE_TESTSUITE_INSERT(Class, Module, Library, false, Priority)
|
|
#define BEAST_DEFINE_TESTSUITE_MANUAL_PRIO(Class, Module, Library, Priority) \
|
|
BEAST_DEFINE_TESTSUITE_INSERT(Class, Module, Library, true, Priority)
|
|
#endif
|
|
|
|
#endif
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
#endif
|