/** * Copyright 2013-2021 Software Radio Systems Limited * * This file is part of srsRAN. * * srsRAN is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License as * published by the Free Software Foundation, either version 3 of * the License, or (at your option) any later version. * * srsRAN is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Affero General Public License for more details. * * A copy of the GNU Affero General Public License can be found in * the LICENSE file in the top-level directory of this distribution * and at http://www.gnu.org/licenses/. * */ #ifndef SRSRAN_TEST_COMMON_H #define SRSRAN_TEST_COMMON_H #include "srsran/config.h" #include "srsran/support/srsran_test.h" #ifdef __cplusplus #include "srsran/common/buffer_pool.h" #include "srsran/common/crash_handler.h" #include "srsran/common/standard_streams.h" #include "srsran/srslog/srslog.h" #include #include namespace srsran { /// This custom sink intercepts log messages to count error and warning log entries. class log_sink_spy : public srslog::sink { public: explicit log_sink_spy(std::unique_ptr f) : srslog::sink(std::move(f)), s(srslog::get_default_sink()) { error_counter.store(0); warning_counter.store(0); } /// Identifier of this custom sink. static const char* name() { return "log_sink_spy"; } /// Returns the number of log entries tagged as errors. unsigned get_error_counter() const { // Flush to make sure all entries have been processed by the backend. srslog::flush(); return error_counter.load(); } /// Returns the number of log entries tagged as warnings. unsigned get_warning_counter() const { // Flush to make sure all entries have been processed by the backend. srslog::flush(); return warning_counter.load(); } /// Resets the counters back to 0. void reset_counters() { // Flush to make sure all entries have been processed by the backend. srslog::flush(); error_counter.store(0); warning_counter.store(0); } srslog::detail::error_string write(srslog::detail::memory_buffer buffer) override { std::string entry(buffer.data(), buffer.size()); if (entry.find("[E]") != std::string::npos) { error_counter.fetch_add(1); } else if (entry.find("[W]") != std::string::npos) { warning_counter.fetch_add(1); } return s.write(buffer); } srslog::detail::error_string flush() override { return s.flush(); } private: srslog::sink& s; std::atomic error_counter; std::atomic warning_counter; }; /// This custom sink intercepts log messages allowing users to check if a certain log entry has been generated. /// Calling spy.has_message("something") will return true if any log entries generated so far contain the string /// "something". /// The log entries history can be cleared with reset so old entries can be discarded. class log_sink_message_spy : public srslog::sink { public: explicit log_sink_message_spy(std::unique_ptr f) : srslog::sink(std::move(f)), s(srslog::get_default_sink()) {} /// Identifier of this custom sink. static const char* name() { return "log_sink_message_spy"; } /// Discards all registered log entries. void reset() { // Flush to make sure all entries have been processed by the backend. srslog::flush(); entries.clear(); } /// Returns true if the string in msg is found in the registered log entries. bool has_message(const std::string& msg) const { srslog::flush(); return std::find_if(entries.cbegin(), entries.cend(), [&](const std::string& entry) { return entry.find(msg) != std::string::npos; }) != entries.cend(); } srslog::detail::error_string write(srslog::detail::memory_buffer buffer) override { entries.emplace_back(buffer.data(), buffer.size()); return s.write(buffer); } srslog::detail::error_string flush() override { return s.flush(); } private: srslog::sink& s; std::vector entries; }; inline void test_init(int argc, char** argv) { // Setup a backtrace pretty printer srsran_debug_handle_crash(argc, argv); srslog::fetch_basic_logger("ALL").set_level(srslog::basic_levels::info); srslog::fetch_basic_logger("TEST").set_level(srslog::basic_levels::info); // Start the log backend. srslog::init(); } inline void copy_msg_to_buffer(unique_byte_buffer_t& pdu, const_byte_span msg) { pdu = srsran::make_byte_buffer(); if (pdu == nullptr) { srslog::fetch_basic_logger("ALL").error("Couldn't allocate PDU in %s().", __FUNCTION__); return; } memcpy(pdu->msg, msg.data(), msg.size()); pdu->N_bytes = msg.size(); } class test_delimit_logger { const size_t delimiter_length = 128; public: template explicit test_delimit_logger(const char* test_name_fmt, Args&&... args) { test_name = fmt::format(test_name_fmt, std::forward(args)...); std::string name_str = fmt::format("[ Test \"{}\" ]", test_name); double nof_repeats = (delimiter_length - name_str.size()) / 2.0; fmt::print("{0:=>{1}}{2}{0:=>{3}}\n", "", (int)floor(nof_repeats), name_str, (int)ceil(nof_repeats)); } test_delimit_logger(const test_delimit_logger&) = delete; test_delimit_logger(test_delimit_logger&&) = delete; test_delimit_logger& operator=(const test_delimit_logger&) = delete; test_delimit_logger& operator=(test_delimit_logger&&) = delete; ~test_delimit_logger() { srslog::flush(); fmt::print("{:=>{}}\n", "", delimiter_length); } private: std::string test_name; }; } // namespace srsran #define CONDERROR(cond, fmt, ...) srsran_assert(not(cond), fmt, ##__VA_ARGS__) #define TESTERROR(fmt, ...) CONDERROR(true, fmt, ##__VA_ARGS__) #endif // __cplusplus #endif // SRSRAN_TEST_COMMON_H