mirror of
https://github.com/XRPLF/rippled.git
synced 2025-11-19 18:45:52 +00:00
* Better include path support in the VSProject scons tool. * Various manual fixes to include paths.
290 lines
7.9 KiB
C++
290 lines
7.9 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_SUITE_H_INCLUDED
|
|
#define BEAST_UNIT_TEST_SUITE_H_INCLUDED
|
|
|
|
#include <beast/unit_test/runner.h>
|
|
|
|
#include <beast/utility/noexcept.h>
|
|
#include <string>
|
|
|
|
namespace beast {
|
|
namespace unit_test {
|
|
|
|
/** 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:
|
|
// 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 "suite aborted on failed condition";
|
|
}
|
|
};
|
|
|
|
// Memberspace
|
|
class log_t
|
|
{
|
|
private:
|
|
friend class suite;
|
|
runner* m_runner;
|
|
|
|
runner*
|
|
operator-> ()
|
|
{
|
|
return m_runner;
|
|
}
|
|
|
|
public:
|
|
log_t ()
|
|
: m_runner (nullptr)
|
|
{
|
|
}
|
|
|
|
template <class T>
|
|
abstract_ostream::scoped_stream_type
|
|
operator<< (T const& t)
|
|
{
|
|
return m_runner->stream() << t;
|
|
}
|
|
|
|
/** Returns the raw stream used for output. */
|
|
abstract_ostream&
|
|
stream()
|
|
{
|
|
return m_runner->stream();
|
|
}
|
|
};
|
|
|
|
bool m_abort = false;
|
|
|
|
public:
|
|
/** Type for scoped stream logging.
|
|
To use this type, declare a local variable of the type
|
|
on the stack in derived class member function and construct
|
|
it from log.stream();
|
|
|
|
@code
|
|
|
|
scoped_stream ss (log.stream();
|
|
|
|
ss << "Hello" << std::endl;
|
|
ss << "world" << std::endl;
|
|
|
|
@endcode
|
|
|
|
Streams constructed in this fashion will not have the line
|
|
ending automatically appended.
|
|
|
|
Thread safety:
|
|
|
|
The scoped_stream may only be used by one thread.
|
|
Multiline output sent to the stream will be atomically
|
|
written to the underlying abstract_Ostream
|
|
*/
|
|
typedef abstract_ostream::scoped_stream_type scoped_stream;
|
|
|
|
/** Memberspace for logging. */
|
|
log_t log;
|
|
|
|
/** 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.
|
|
*/
|
|
void
|
|
operator() (runner& r)
|
|
{
|
|
log.m_runner = &r;
|
|
|
|
try
|
|
{
|
|
run();
|
|
}
|
|
catch (abort_exception const&)
|
|
{
|
|
// ends the suite
|
|
}
|
|
catch (std::exception const& e)
|
|
{
|
|
fail (std::string ("unhandled exception: ") +
|
|
std::string (e.what()));
|
|
}
|
|
catch (...)
|
|
{
|
|
fail ("unhandled exception");
|
|
}
|
|
}
|
|
|
|
enum abort_t
|
|
{
|
|
no_abort_on_fail,
|
|
abort_on_fail
|
|
};
|
|
|
|
/** 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 If `true`, the suite will be stopped on first failure.
|
|
*/
|
|
void
|
|
testcase (std::string const& name,
|
|
abort_t abort = no_abort_on_fail)
|
|
{
|
|
m_abort = abort == abort_on_fail;
|
|
log->testcase (name);
|
|
}
|
|
|
|
/** Evaluate a test condition.
|
|
The condition is passed as a template argument instead of `bool` so
|
|
that implicit conversion is not required. The `reason` argument is
|
|
logged if the condition is false.
|
|
@return `true` if the test condition indicates success.
|
|
*/
|
|
template <class Condition>
|
|
bool
|
|
expect (Condition shouldBeTrue, std::string const& reason = "")
|
|
{
|
|
if (shouldBeTrue)
|
|
pass();
|
|
else
|
|
fail (reason);
|
|
return shouldBeTrue;
|
|
}
|
|
|
|
// DEPRECATED
|
|
// @return `true` if the test condition indicates success (a false value)
|
|
template <class Condition>
|
|
bool
|
|
unexpected (Condition shouldBeFalse, std::string const& reason = "")
|
|
{
|
|
if (! shouldBeFalse)
|
|
pass();
|
|
else
|
|
fail (reason);
|
|
return ! shouldBeFalse;
|
|
}
|
|
|
|
/** Record a successful test condition. */
|
|
void
|
|
pass()
|
|
{
|
|
log->pass();
|
|
}
|
|
|
|
/** Record a failure. */
|
|
void
|
|
fail (std::string const& reason= "")
|
|
{
|
|
log->fail (reason);
|
|
if (m_abort)
|
|
throw abort_exception();
|
|
}
|
|
|
|
private:
|
|
/** Runs the suite. */
|
|
virtual
|
|
void
|
|
run() = 0;
|
|
};
|
|
|
|
} // unit_test
|
|
} // beast
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
// detail:
|
|
// This inserts the suite with the given manual flag
|
|
#define BEAST_DEFINE_TESTSUITE_INSERT(Class,Module,Library,manual) \
|
|
static beast::unit_test::detail::global_suite_instance <Class##_test> \
|
|
Library ## Module ## Class ## _test_instance ( \
|
|
#Class, #Module, #Library, manual);
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
// 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.
|
|
*/
|
|
|
|
#if BEAST_NO_UNIT_TEST_INLINE
|
|
#define BEAST_DEFINE_TESTSUITE(Class,Module,Library)
|
|
|
|
#else
|
|
#include <beast/unit_test/global_suites.h>
|
|
#define BEAST_DEFINE_TESTSUITE(Class,Module,Library) \
|
|
BEAST_DEFINE_TESTSUITE_INSERT(Class,Module,Library,false)
|
|
#define BEAST_DEFINE_TESTSUITE_MANUAL(Class,Module,Library) \
|
|
BEAST_DEFINE_TESTSUITE_INSERT(Class,Module,Library,true)
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
#endif
|