123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219 |
- /*
- * Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
- * Copyright (c) 2021, Andrew Kaster <akaster@serenityos.org>
- *
- * SPDX-License-Identifier: BSD-2-Clause
- */
- #pragma once
- #include <AK/Assertions.h>
- #include <AK/CheckedFormatString.h>
- #include <AK/Math.h>
- #include <AK/SourceLocation.h>
- #include <LibTest/CrashTest.h>
- #include <LibTest/Randomized/RandomnessSource.h>
- #include <LibTest/TestResult.h>
- namespace Test {
- // Declare helpers so that we can call them from VERIFY in included headers
- // the setter for TestResult is already declared in TestResult.h
- TestResult current_test_result();
- Randomized::RandomnessSource& randomness_source();
- void set_randomness_source(Randomized::RandomnessSource);
- bool is_reporting_enabled();
- void enable_reporting();
- void disable_reporting();
- u64 randomized_runs();
- template<typename T>
- void expect(T const& expression, StringView expression_string, SourceLocation location = SourceLocation::current())
- {
- if (!static_cast<bool>(expression)) {
- if (is_reporting_enabled())
- warnln("\033[31;1mFAIL\033[0m: {}:{}: EXPECT({}) failed", location.filename(), location.line_number(), expression_string);
- set_current_test_result(TestResult::Failed);
- }
- }
- template<typename LHS, typename RHS>
- void expect_equality(LHS const& lhs, RHS const& rhs, StringView lhs_string, StringView rhs_string, SourceLocation location = SourceLocation::current())
- {
- if (lhs != rhs) {
- if (is_reporting_enabled()) {
- warnln("\033[31;1mFAIL\033[0m: {}:{}: EXPECT_EQ({}, {}) failed with lhs={} and rhs={}",
- location.filename(), location.line_number(), lhs_string, rhs_string,
- FormatIfSupported { lhs }, FormatIfSupported { rhs });
- }
- set_current_test_result(TestResult::Failed);
- }
- }
- template<typename LHS, typename RHS>
- void expect_truthy_equality(LHS const& lhs, RHS const& rhs, StringView lhs_string, StringView rhs_string, SourceLocation location = SourceLocation::current())
- {
- if (static_cast<bool>(lhs) != static_cast<bool>(rhs)) {
- if (is_reporting_enabled()) {
- warnln("\033[31;1mFAIL\033[0m: {}:{}: EXPECT_EQ_TRUTH({}, {}) failed with lhs={} ({}) and rhs={} ({})",
- location.filename(), location.line_number(), lhs_string, rhs_string,
- FormatIfSupported { lhs }, static_cast<bool>(lhs),
- FormatIfSupported { rhs }, static_cast<bool>(rhs));
- }
- set_current_test_result(TestResult::Failed);
- }
- }
- template<typename LHS, typename RHS>
- void expect_equality_with_forced_logging(LHS const& lhs, RHS const& rhs, StringView lhs_string, StringView rhs_string, SourceLocation location = SourceLocation::current())
- {
- if (lhs != rhs) {
- if (is_reporting_enabled()) {
- warnln("\033[31;1mFAIL\033[0m: {}:{}: EXPECT_EQ({}, {}) failed with lhs={} and rhs={}",
- location.filename(), location.line_number(), lhs_string, rhs_string,
- lhs, rhs);
- }
- set_current_test_result(TestResult::Failed);
- }
- }
- template<typename LHS, typename RHS>
- void expect_inequality(LHS const& lhs, RHS const& rhs, StringView lhs_string, StringView rhs_string, SourceLocation location = SourceLocation::current())
- {
- if (lhs == rhs) {
- if (is_reporting_enabled()) {
- warnln("\033[31;1mFAIL\033[0m: {}:{}: EXPECT_NE({}, {}) failed with lhs={} and rhs={}",
- location.filename(), location.line_number(), lhs_string, rhs_string,
- FormatIfSupported { lhs }, FormatIfSupported { rhs });
- }
- set_current_test_result(TestResult::Failed);
- }
- }
- template<FloatingPoint LHS, FloatingPoint RHS>
- void expect_approximate(LHS lhs, RHS rhs, StringView lhs_string, StringView rhs_string, double tolerance, SourceLocation location = SourceLocation::current())
- {
- auto diff = static_cast<double>(lhs) - static_cast<double>(rhs);
- if (AK::fabs(diff) > tolerance) {
- if (is_reporting_enabled()) {
- warnln("\033[31;1mFAIL\033[0m: {}:{}: EXPECT_APPROXIMATE({}, {}) failed with lhs={} and rhs={}, (lhs-rhs)={}",
- location.filename(), location.line_number(), lhs_string, rhs_string,
- lhs, rhs, diff);
- }
- set_current_test_result(TestResult::Failed);
- }
- }
- template<typename T>
- bool assume(T const& expression, StringView expression_string, SourceLocation location = SourceLocation::current())
- {
- if (!static_cast<bool>(expression)) {
- if (is_reporting_enabled()) {
- warnln("\033[31;1mREJECTED\033[0m: {}:{}: Couldn't generate random value satisfying ASSUME({})",
- location.filename(), location.line_number(), expression_string);
- }
- set_current_test_result(TestResult::Rejected);
- return false;
- }
- return true;
- }
- }
- #define EXPECT(x) \
- do { \
- ::Test::expect(x, #x##sv); \
- } while (false)
- #define EXPECT_EQ(a, b) \
- do { \
- ::Test::expect_equality(a, b, #a##sv, #b##sv); \
- } while (false)
- #define EXPECT_EQ_TRUTH(a, b) \
- do { \
- ::Test::expect_truthy_equality(a, b, #a##sv, #b##sv); \
- } while (false)
- // If you're stuck and `EXPECT_EQ` seems to refuse to print anything useful,
- // try this: It'll spit out a nice compiler error telling you why it doesn't print.
- #define EXPECT_EQ_FORCE(a, b) \
- do { \
- ::Test::expect_equality_with_forced_logging(a, b, #a##sv, #b##sv); \
- } while (false)
- #define EXPECT_NE(a, b) \
- do { \
- ::Test::expect_inequality(a, b, #a##sv, #b##sv); \
- } while (false)
- #define EXPECT_APPROXIMATE_WITH_ERROR(a, b, err) \
- do { \
- ::Test::expect_approximate(a, b, #a##sv, #b##sv, err); \
- } while (false)
- #define EXPECT_APPROXIMATE(a, b) EXPECT_APPROXIMATE_WITH_ERROR(a, b, 0.0000005)
- #define ASSUME(x) \
- do { \
- if (!::Test::assume(x, #x##sv)) \
- return; \
- } while (false)
- #define FAIL(message) \
- do { \
- if (::Test::is_reporting_enabled()) \
- ::AK::warnln("\033[31;1mFAIL\033[0m: {}:{}: {}", __FILE__, __LINE__, message); \
- ::Test::set_current_test_result(::Test::TestResult::Failed); \
- } while (false)
- // To use, specify the lambda to execute in a sub process and verify it exits:
- // EXPECT_CRASH("This should fail", []{
- // return Test::Crash::Failure::DidNotCrash;
- // });
- #define EXPECT_CRASH(test_message, test_func) \
- do { \
- Test::Crash crash(test_message, test_func); \
- if (!crash.run()) \
- ::Test::set_current_test_result(::Test::TestResult::Failed); \
- } while (false)
- #define EXPECT_CRASH_WITH_SIGNAL(test_message, signal, test_func) \
- do { \
- Test::Crash crash(test_message, test_func, (signal)); \
- if (!crash.run()) \
- ::Test::set_current_test_result(::Test::TestResult::Failed); \
- } while (false)
- #define EXPECT_NO_CRASH(test_message, test_func) \
- do { \
- Test::Crash crash(test_message, test_func, 0); \
- if (!crash.run()) \
- ::Test::set_current_test_result(::Test::TestResult::Failed); \
- } while (false)
- #define TRY_OR_FAIL(expression) \
- ({ \
- /* Ignore -Wshadow to allow nesting the macro. */ \
- AK_IGNORE_DIAGNOSTIC("-Wshadow", \
- auto&& _temporary_result = (expression)); \
- static_assert(!::AK::Detail::IsLvalueReference<decltype(_temporary_result.release_value())>, \
- "Do not return a reference from a fallible expression"); \
- if (_temporary_result.is_error()) [[unlikely]] { \
- FAIL(_temporary_result.release_error()); \
- return; \
- } \
- _temporary_result.release_value(); \
- })
|