//////////////////////////////////////////////////////////////////////////////// // // Copyright 2006 - 2021, Tomas Babej, Paul Beckingham, Federico Hernandez. // // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to deal // in the Software without restriction, including without limitation the rights // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell // copies of the Software, and to permit persons to whom the Software is // furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included // in all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS // OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL // THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE // SOFTWARE. // // https://www.opensource.org/licenses/mit-license.php // //////////////////////////////////////////////////////////////////////////////// #include #include #include #include #include #include #include #include /////////////////////////////////////////////////////////////////////////////// UnitTest::UnitTest() : _planned(0), _counter(0), _passed(0), _failed(0), _skipped(0) {} /////////////////////////////////////////////////////////////////////////////// UnitTest::UnitTest(int planned) : _planned(planned), _counter(0), _passed(0), _failed(0), _skipped(0) { std::cout << "1.." << _planned << '\n'; } /////////////////////////////////////////////////////////////////////////////// UnitTest::~UnitTest() { float percentPassed = 0.0; if (_planned > 0) percentPassed = (100.0 * _passed) / std::max(_planned, _passed + _failed + _skipped); if (_counter < _planned) { std::cout << "# Only " << _counter << " tests, out of a planned " << _planned << " were run.\n"; _skipped += _planned - _counter; } else if (_counter > _planned) std::cout << "# " << _counter << " tests were run, but only " << _planned << " were planned.\n"; std::cout << "# " << _passed << " passed, " << _failed << " failed, " << _skipped << " skipped. " << std::setprecision(3) << percentPassed << "% passed.\n"; exit(_failed > 0); } /////////////////////////////////////////////////////////////////////////////// void UnitTest::plan(int planned) { _planned = planned; _counter = 0; _passed = 0; _failed = 0; _skipped = 0; std::cout << "1.." << _planned << '\n'; } /////////////////////////////////////////////////////////////////////////////// void UnitTest::planMore(int extra) { _planned += extra; std::cout << "1.." << _planned << '\n'; } /////////////////////////////////////////////////////////////////////////////// void UnitTest::ok(bool expression, const std::string& name, bool expfail /* = false */) { ++_counter; bool success = expression; if (success and !expfail) { ++_passed; std::cout << green("ok") << " " << _counter << " - " << name << '\n'; } else { if (success == expfail) ++_failed; std::cout << red("not ok") << " " << _counter << " - " << name << (expfail ? (success ? " # FIXED" : " # TODO") : "") << '\n'; } } /////////////////////////////////////////////////////////////////////////////// void UnitTest::notok(bool expression, const std::string& name, bool expfail /* = false */) { ++_counter; bool success = not expression; if (success and !expfail) { ++_passed; std::cout << green("ok") << " " << _counter << " - " << name << '\n'; } else { if (success == expfail) ++_failed; std::cout << red("not ok") << " " << _counter << " - " << name << (expfail ? (success ? " # FIXED" : " # TODO") : "") << '\n'; } } /////////////////////////////////////////////////////////////////////////////// void UnitTest::is(bool actual, bool expected, const std::string& name, bool expfail /* = false */) { ++_counter; bool success = (actual == expected); if (success and !expfail) { ++_passed; std::cout << green("ok") << " " << _counter << " - " << name << '\n'; } else { if (success == expfail) ++_failed; std::cout << red("not ok") << " " << _counter << " - " << name << (expfail ? (success ? " # FIXED" : " # TODO") : "") << "\n# expected: " << expected << "\n# got: " << actual << '\n'; } } /////////////////////////////////////////////////////////////////////////////// void UnitTest::is(size_t actual, size_t expected, const std::string& name, bool expfail /* = false */) { ++_counter; bool success = (actual == expected); if (success and !expfail) { ++_passed; std::cout << green("ok") << " " << _counter << " - " << name << '\n'; } else { if (success == expfail) ++_failed; std::cout << red("not ok") << " " << _counter << " - " << name << (expfail ? (success ? " # FIXED" : " # TODO") : "") << "\n# expected: " << expected << "\n# got: " << actual << '\n'; } } /////////////////////////////////////////////////////////////////////////////// void UnitTest::is(int actual, int expected, const std::string& name, bool expfail /* = false */) { ++_counter; bool success = (actual == expected); if (success and !expfail) { ++_passed; std::cout << green("ok") << " " << _counter << " - " << name << '\n'; } else { if (success == expfail) ++_failed; std::cout << red("not ok") << " " << _counter << " - " << name << (expfail ? (success ? " # FIXED" : " # TODO") : "") << "\n# expected: " << expected << "\n# got: " << actual << '\n'; } } /////////////////////////////////////////////////////////////////////////////// void UnitTest::is(double actual, double expected, const std::string& name, bool expfail /* = false */) { ++_counter; bool success = (actual == expected); if (success and !expfail) { ++_passed; std::cout << green("ok") << " " << _counter << " - " << name << '\n'; } else { if (success == expfail) ++_failed; std::cout << red("not ok") << " " << _counter << " - " << name << (expfail ? (success ? " # FIXED" : " # TODO") : "") << "\n# expected: " << expected << "\n# got: " << actual << '\n'; } } /////////////////////////////////////////////////////////////////////////////// void UnitTest::is(double actual, double expected, double tolerance, const std::string& name, bool expfail /* = false */) { ++_counter; bool success = (fabs(actual - expected) <= tolerance); if (success and !expfail) { ++_passed; std::cout << green("ok") << " " << _counter << " - " << name << '\n'; } else { if (success == expfail) ++_failed; std::cout << red("not ok") << " " << _counter << " - " << name << (expfail ? (success ? " # FIXED" : " # TODO") : "") << "\n# expected: " << expected << "\n# got: " << actual << '\n'; } } /////////////////////////////////////////////////////////////////////////////// void UnitTest::is(unsigned char actual, unsigned char expected, const std::string& name, bool expfail /* = false */) { ++_counter; bool success = (actual == expected); if (success and !expfail) { ++_passed; std::cout << green("ok") << " " << _counter << " - " << name << '\n'; } else { if (success == expfail) ++_failed; std::cout << red("not ok") << " " << _counter << " - " << name << (expfail ? (success ? " # FIXED" : " # TODO") : "") << "\n# expected: " << expected << "\n# got: " << actual << '\n'; } } /////////////////////////////////////////////////////////////////////////////// void UnitTest::is(const std::string& actual, const std::string& expected, const std::string& name, bool expfail /* = false */) { ++_counter; bool success = (actual == expected); if (success and !expfail) { ++_passed; std::cout << green("ok") << " " << _counter << " - " << name << '\n'; } else { if (success == expfail) ++_failed; std::cout << red("not ok") << " " << _counter << " - " << name << (expfail ? (success ? " # FIXED" : " # TODO") : "") << "\n# expected: '" << expected << "'" << "\n# got: '" << actual << "'\n"; } } /////////////////////////////////////////////////////////////////////////////// void UnitTest::is(const char* actual, const char* expected, const std::string& name, bool expfail /* = false */) { ++_counter; bool success = (!strcmp(actual, expected)); if (success and !expfail) { ++_passed; std::cout << green("ok") << " " << _counter << " - " << name << '\n'; } else { if (success == expfail) ++_failed; std::cout << red("not ok") << " " << _counter << " - " << name << (expfail ? (success ? " # FIXED" : " # TODO") : "") << "\n# expected: '" << expected << "'" << "\n# got: '" << actual << "'\n"; } } /////////////////////////////////////////////////////////////////////////////// void UnitTest::diag(const std::string& text) { auto start = text.find_first_not_of(" \t\n\r\f"); auto end = text.find_last_not_of(" \t\n\r\f"); if (start != std::string::npos && end != std::string::npos) std::cout << "# " << text.substr(start, end - start + 1) << '\n'; } /////////////////////////////////////////////////////////////////////////////// void UnitTest::pass(const std::string& text) { ++_counter; ++_passed; std::cout << green("ok") << " " << _counter << " - " << text << '\n'; } /////////////////////////////////////////////////////////////////////////////// void UnitTest::fail(const std::string& text) { ++_counter; ++_failed; std::cout << red("not ok") << " " << _counter << " - " << text << '\n'; } /////////////////////////////////////////////////////////////////////////////// void UnitTest::skip(const std::string& text) { ++_counter; ++_skipped; std::cout << yellow("ok") << " " << _counter << " - " << text << " # skip" << '\n'; } /////////////////////////////////////////////////////////////////////////////// std::string UnitTest::red(const std::string& input) { if (isatty(fileno(stdout))) return std::string("\033[31m" + input + "\033[0m"); return input; } /////////////////////////////////////////////////////////////////////////////// std::string UnitTest::green(const std::string& input) { if (isatty(fileno(stdout))) return std::string("\033[32m" + input + "\033[0m"); return input; } /////////////////////////////////////////////////////////////////////////////// std::string UnitTest::yellow(const std::string& input) { if (isatty(fileno(stdout))) return std::string("\033[33m" + input + "\033[0m"); return input; } ///////////////////////////////////////////////////////////////////////////////