From 99f7c532466a2c559a5c82df0c495203c3bb3950 Mon Sep 17 00:00:00 2001 From: Artur Ryt Date: Wed, 21 Jun 2023 21:46:24 +0200 Subject: [PATCH] Explicit argc/argv passing This allows user to explicitly specify args for the run. ```c++ int main(int argc, const char** argv) { // explicitly run registered test suites and manually pass argc/argv return ut::cfg<>.run({.argc = argc, .argv = argv}); } ``` To get it working yet another event has been added: `events::run_begin`, which contains `argc`/`argv` passed by the user. --- example/cfg/reporter.cpp | 1 + example/run.cpp | 5 +++-- include/boost/ut.hpp | 33 +++++++++++++++++++++++++-------- test/ut/ut.cpp | 19 +++++++++++++++++++ 4 files changed, 48 insertions(+), 10 deletions(-) diff --git a/example/cfg/reporter.cpp b/example/cfg/reporter.cpp index 95055c4d..09155450 100644 --- a/example/cfg/reporter.cpp +++ b/example/cfg/reporter.cpp @@ -26,6 +26,7 @@ class reporter { auto on(ut::events::fatal_assertion) -> void {} auto on(ut::events::exception) -> void {} auto on(ut::events::summary) -> void {} + auto on(ut::events::run_begin) -> void {} }; } // namespace cfg diff --git a/example/run.cpp b/example/run.cpp index 329ba428..80ea2bef 100644 --- a/example/run.cpp +++ b/example/run.cpp @@ -17,6 +17,7 @@ ut::suite _ = [] { }; }; -int main() { - return ut::cfg<>.run(); // explicitly run registered test suites +int main(int argc, const char** argv) { + // explicitly run registered test suites and manually pass argc/argv + return ut::cfg<>.run({.argc = argc, .argv = argv}); } diff --git a/include/boost/ut.hpp b/include/boost/ut.hpp index b4dbef74..3732a813 100644 --- a/include/boost/ut.hpp +++ b/include/boost/ut.hpp @@ -544,6 +544,10 @@ fixed_string(const CharT (&str)[N]) -> fixed_string; struct none {}; namespace events { +struct run_begin { + int argc{}; + const char** argv{}; +}; struct test_begin { std::string_view type{}; std::string_view name{}; @@ -752,11 +756,17 @@ struct cfg { std::cout << "version: " << BOOST_UT_VERSION << std::endl; } + static inline void parse_arg_with_fallback(int argc, const char* argv[]) { + if (argc > 0 && argv != nullptr) { + cfg::largc = argc; + cfg::largv = argv; + } + parse(cfg::largc, cfg::largv); + } + static inline void parse(int argc, const char* argv[]) { const std::size_t n_args = static_cast(argc); if (n_args > 0 && argv != nullptr) { - cfg::largc = argc; - cfg::largv = argv; executable_name = argv[0]; } query_pattern = ""; @@ -766,15 +776,15 @@ struct cfg { auto cmd_option = find_arg(cmd); if (!cmd_option.has_value()) { if (found_first_option) { - std::cerr << "unknown option: '" << argv[i] << "' run:" << std::endl; - std::cerr << "'" << argv[0] << " --help'" << std::endl; + std::cerr << "unknown option: '" << cmd << "' run:" << std::endl; + std::cerr << "'" << executable_name << " --help'" << std::endl; std::cerr << "for additional help" << std::endl; std::exit(-1); } else { if (i > 1U) { query_pattern.append(" "); } - query_pattern.append(argv[i]); + query_pattern.append(cmd); } continue; } @@ -1428,6 +1438,8 @@ class reporter { printer_ = static_cast(printer); } + auto on(events::run_begin) -> void {} + auto on(events::test_begin test_begin) -> void { printer_ << "Running \"" << test_begin.name << "\"..."; fails_ = asserts_.fail; @@ -1631,8 +1643,11 @@ class reporter_junit { constexpr auto operator=(TPrinter printer) { printer_ = static_cast(printer); } - reporter_junit() : lcout_(std::cout.rdbuf()) { - ::boost::ut::detail::cfg::parse(detail::cfg::largc, detail::cfg::largv); + reporter_junit() : lcout_(std::cout.rdbuf()) {} + ~reporter_junit() { std::cout.rdbuf(cout_save); } + + auto on(events::run_begin run) { + ::boost::ut::detail::cfg::parse_arg_with_fallback(run.argc, run.argv); if (detail::cfg::show_reporters) { std::cout << "available reporter:\n"; @@ -1652,7 +1667,6 @@ class reporter_junit { std::cout.rdbuf(ss_out_.rdbuf()); } } - ~reporter_junit() { std::cout.rdbuf(cout_save); } auto on(events::suite_begin suite) -> void { while (active_test_.size() > 0) { @@ -1912,6 +1926,8 @@ struct options { struct run_cfg { bool report_errors{false}; + int argc{0}; + const char** argv{nullptr}; }; template , auto MaxPathSize = 16> @@ -2115,6 +2131,7 @@ class runner { [[nodiscard]] auto run(run_cfg rc = {}) -> bool { run_ = true; + reporter_.on(events::run_begin{.argc = rc.argc, .argv = rc.argv}); for (const auto& [suite, suite_name] : suites_) { // add reporter in/out if constexpr (requires { reporter_.on(events::suite_begin{}); }) { diff --git a/test/ut/ut.cpp b/test/ut/ut.cpp index a5240b7a..3420558b 100644 --- a/test/ut/ut.cpp +++ b/test/ut/ut.cpp @@ -282,13 +282,22 @@ struct test_summary_reporter : ut::reporter { summary_counter_ = &counter; } + auto count_runs(std::size_t& counter) -> void { runs_counter_ = &counter; } + auto on(ut::events::summary) -> void { if (summary_counter_) { ++*summary_counter_; } } + auto on(ut::events::run_begin) -> void { + if (runs_counter_) { + ++*runs_counter_; + } + } + std::size_t* summary_counter_{}; + std::size_t* runs_counter_{}; }; struct test_summary_runner : ut::runner { @@ -826,6 +835,16 @@ int main() { test_assert(1 == summary_count); } + { + std::size_t run_count = 0; + { + auto run = test_summary_runner{}; + run.reporter_.count_runs(run_count); + test_assert(false == run.run({.report_errors = true})); + } + test_assert(1 == run_count); + } + auto& test_cfg = ut::cfg; {