#pragma once #include #include #include #include #include #include #define TEST_SEQUENCE(...) static auto ANONYMOUS_VARIABLE(TEST_SEQUENCE) = ::hex::test::TestSequenceExecutor(__VA_ARGS__) + []() -> int #define TEST_FAIL() return EXIT_FAILURE #define TEST_SUCCESS() return EXIT_SUCCESS #define FAILING true #define TEST_ASSERT(x) return (x) ? EXIT_SUCCESS : EXIT_FAILURE namespace hex::test { struct Test { std::function function; bool shouldFail; }; class Tests { public: static auto addTest(const std::string &name, const std::function &func, bool shouldFail) noexcept { s_tests.insert({ name, { func, shouldFail } }); return 0; } static auto& get()noexcept { return s_tests; } private: static inline std::map s_tests; }; template class TestSequence { public: TestSequence(const std::string& name, F func, bool shouldFail) noexcept { Tests::addTest(name, func, shouldFail); } TestSequence& operator=(TestSequence &&) = delete; }; struct TestSequenceExecutor { explicit TestSequenceExecutor(std::string name, bool shouldFail = false) noexcept : m_name(std::move(name)), m_shouldFail(shouldFail) { } [[nodiscard]] const auto& getName() const noexcept { return this->m_name; } [[nodiscard]] bool shouldFail() const noexcept { return this->m_shouldFail; } private: std::string m_name; bool m_shouldFail; }; template TestSequence operator+(TestSequenceExecutor executor, F&& f) noexcept { return TestSequence(executor.getName(), std::forward(f), executor.shouldFail()); } }