2019-02-25 14:48:22 +01:00
|
|
|
/* Icinga 2 | (c) 2012 Icinga GmbH | GPLv2+ */
|
2019-02-22 16:58:26 +01:00
|
|
|
|
|
|
|
#include "base/utility.hpp"
|
|
|
|
#include <chrono>
|
|
|
|
#include <BoostTestTargetConfig.h>
|
|
|
|
|
2021-01-28 16:25:12 +01:00
|
|
|
#ifdef _WIN32
|
|
|
|
# include <windows.h>
|
|
|
|
# include <shellapi.h>
|
|
|
|
#endif /* _WIN32 */
|
|
|
|
|
2019-02-22 16:58:26 +01:00
|
|
|
using namespace icinga;
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_SUITE(base_utility)
|
|
|
|
|
2019-08-14 11:22:55 +02:00
|
|
|
BOOST_AUTO_TEST_CASE(parse_version)
|
|
|
|
{
|
|
|
|
BOOST_CHECK(Utility::ParseVersion("2.11.0-0.rc1.1") == "2.11.0");
|
|
|
|
BOOST_CHECK(Utility::ParseVersion("v2.10.5") == "2.10.5");
|
|
|
|
BOOST_CHECK(Utility::ParseVersion("r2.11.1") == "2.11.1");
|
|
|
|
BOOST_CHECK(Utility::ParseVersion("v2.11.0-rc1-58-g7c1f716da") == "2.11.0");
|
|
|
|
|
|
|
|
BOOST_CHECK(Utility::ParseVersion("v2.11butactually3.0") == "v2.11butactually3.0");
|
|
|
|
}
|
|
|
|
|
2019-08-14 13:14:43 +02:00
|
|
|
BOOST_AUTO_TEST_CASE(compare_version)
|
|
|
|
{
|
|
|
|
BOOST_CHECK(Utility::CompareVersion("2.10.5", Utility::ParseVersion("v2.10.4")) < 0);
|
|
|
|
BOOST_CHECK(Utility::CompareVersion("2.11.0", Utility::ParseVersion("2.11.0-0")) == 0);
|
|
|
|
BOOST_CHECK(Utility::CompareVersion("2.10.5", Utility::ParseVersion("2.11.0-0.rc1.1")) > 0);
|
|
|
|
}
|
|
|
|
|
2019-02-22 16:58:26 +01:00
|
|
|
BOOST_AUTO_TEST_CASE(comparepasswords_works)
|
|
|
|
{
|
|
|
|
BOOST_CHECK(Utility::ComparePasswords("", ""));
|
|
|
|
|
|
|
|
BOOST_CHECK(!Utility::ComparePasswords("x", ""));
|
|
|
|
BOOST_CHECK(!Utility::ComparePasswords("", "x"));
|
|
|
|
|
|
|
|
BOOST_CHECK(Utility::ComparePasswords("x", "x"));
|
|
|
|
BOOST_CHECK(!Utility::ComparePasswords("x", "y"));
|
|
|
|
|
|
|
|
BOOST_CHECK(Utility::ComparePasswords("abcd", "abcd"));
|
|
|
|
BOOST_CHECK(!Utility::ComparePasswords("abc", "abcd"));
|
|
|
|
BOOST_CHECK(!Utility::ComparePasswords("abcde", "abcd"));
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(comparepasswords_issafe)
|
|
|
|
{
|
|
|
|
using std::chrono::duration_cast;
|
|
|
|
using std::chrono::microseconds;
|
|
|
|
using std::chrono::steady_clock;
|
|
|
|
|
|
|
|
String a, b;
|
|
|
|
|
|
|
|
a.Append(200000001, 'a');
|
|
|
|
b.Append(200000002, 'b');
|
|
|
|
|
|
|
|
auto start1 (steady_clock::now());
|
|
|
|
|
|
|
|
Utility::ComparePasswords(a, a);
|
|
|
|
|
|
|
|
auto duration1 (steady_clock::now() - start1);
|
|
|
|
|
|
|
|
auto start2 (steady_clock::now());
|
|
|
|
|
|
|
|
Utility::ComparePasswords(a, b);
|
|
|
|
|
|
|
|
auto duration2 (steady_clock::now() - start2);
|
|
|
|
|
|
|
|
double diff = (double)duration_cast<microseconds>(duration1).count() / (double)duration_cast<microseconds>(duration2).count();
|
2019-02-26 11:45:03 +01:00
|
|
|
BOOST_WARN(0.9 <= diff && diff <= 1.1);
|
2019-02-22 16:58:26 +01:00
|
|
|
}
|
|
|
|
|
2019-03-13 18:12:58 +01:00
|
|
|
BOOST_AUTO_TEST_CASE(validateutf8)
|
|
|
|
{
|
|
|
|
BOOST_CHECK(Utility::ValidateUTF8("") == "");
|
|
|
|
BOOST_CHECK(Utility::ValidateUTF8("a") == "a");
|
|
|
|
BOOST_CHECK(Utility::ValidateUTF8("\xC3") == "\xEF\xBF\xBD");
|
|
|
|
BOOST_CHECK(Utility::ValidateUTF8("\xC3\xA4") == "\xC3\xA4");
|
|
|
|
}
|
|
|
|
|
2021-01-28 16:25:12 +01:00
|
|
|
BOOST_AUTO_TEST_CASE(EscapeCreateProcessArg)
|
|
|
|
{
|
|
|
|
#ifdef _WIN32
|
2021-04-09 15:28:40 +02:00
|
|
|
using convert = std::wstring_convert<std::codecvt<wchar_t, char, std::mbstate_t>, wchar_t>;
|
|
|
|
|
2021-01-28 16:25:12 +01:00
|
|
|
std::vector<std::string> testdata = {
|
|
|
|
R"(foobar)",
|
|
|
|
R"(foo bar)",
|
|
|
|
R"(foo"bar)",
|
|
|
|
R"("foo bar")",
|
|
|
|
R"(" \" \\" \\\" \\\\")",
|
|
|
|
R"( !"#$$%&'()*+,-./09:;<=>?@AZ[\]^_`az{|}~ " \" \\" \\\" \\\\")",
|
|
|
|
"'foo\nbar'",
|
|
|
|
};
|
|
|
|
|
|
|
|
for (const auto& t : testdata) {
|
|
|
|
// Prepend some fake exec name as the first argument is handled differently.
|
|
|
|
std::string escaped = "some.exe " + Utility::EscapeCreateProcessArg(t);
|
|
|
|
int argc;
|
2021-04-09 15:28:40 +02:00
|
|
|
std::shared_ptr<LPWSTR> argv(CommandLineToArgvW(convert{}.from_bytes(escaped.c_str()).data(), &argc), LocalFree);
|
2021-01-28 16:25:12 +01:00
|
|
|
BOOST_CHECK_MESSAGE(argv != nullptr, "CommandLineToArgvW() should not return nullptr for " << t);
|
|
|
|
BOOST_CHECK_MESSAGE(argc == 2, "CommandLineToArgvW() should find 2 arguments for " << t);
|
|
|
|
if (argc >= 2) {
|
2021-04-09 15:28:40 +02:00
|
|
|
std::string unescaped = convert{}.to_bytes(argv.get()[1]);
|
2021-01-28 16:25:12 +01:00
|
|
|
BOOST_CHECK_MESSAGE(unescaped == t,
|
|
|
|
"CommandLineToArgvW() should return original value for " << t << " (got: " << unescaped << ")");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif /* _WIN32 */
|
|
|
|
}
|
|
|
|
|
2021-06-17 08:49:54 +02:00
|
|
|
BOOST_AUTO_TEST_CASE(TruncateUsingHash)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* Note: be careful when changing the output of TruncateUsingHash as it is used to derive file names that should not
|
|
|
|
* change between versions or would need special handling if they do (/var/lib/icinga2/api/packages/_api).
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* minimum allowed value for maxLength template parameter */
|
|
|
|
BOOST_CHECK_EQUAL(Utility::TruncateUsingHash<44>(std::string(64, 'a')),
|
|
|
|
"a...0098ba824b5c16427bd7a1122a5a442a25ec644d");
|
|
|
|
|
|
|
|
BOOST_CHECK_EQUAL(Utility::TruncateUsingHash<80>(std::string(100, 'a')),
|
|
|
|
std::string(37, 'a') + "...7f9000257a4918d7072655ea468540cdcbd42e0c");
|
|
|
|
|
|
|
|
/* short enough values should not be truncated */
|
|
|
|
BOOST_CHECK_EQUAL(Utility::TruncateUsingHash<80>(""), "");
|
|
|
|
BOOST_CHECK_EQUAL(Utility::TruncateUsingHash<80>(std::string(60, 'a')), std::string(60, 'a'));
|
|
|
|
BOOST_CHECK_EQUAL(Utility::TruncateUsingHash<80>(std::string(79, 'a')), std::string(79, 'a'));
|
|
|
|
|
|
|
|
/* inputs of maxLength are hashed to avoid collisions */
|
|
|
|
BOOST_CHECK_EQUAL(Utility::TruncateUsingHash<80>(std::string(80, 'a')),
|
|
|
|
std::string(37, 'a') + "...86f33652fcffd7fa1443e246dd34fe5d00e25ffd");
|
|
|
|
}
|
|
|
|
|
2024-08-21 10:55:09 +02:00
|
|
|
BOOST_AUTO_TEST_CASE(FormatDateTime) {
|
|
|
|
using time_t_limit = std::numeric_limits<time_t>;
|
|
|
|
|
|
|
|
// Helper to repeat a given string a number of times.
|
|
|
|
auto repeat = [](const std::string& s, size_t n) {
|
|
|
|
std::ostringstream stream;
|
|
|
|
for (size_t i = 0; i < n; ++i) {
|
|
|
|
stream << s;
|
|
|
|
}
|
|
|
|
return stream.str();
|
|
|
|
};
|
|
|
|
|
|
|
|
// Valid inputs.
|
|
|
|
const double ts = 1136214245.0; // 2006-01-02 15:04:05 UTC
|
|
|
|
BOOST_CHECK_EQUAL("2006-01-02 15:04:05", Utility::FormatDateTime("%F %T", ts));
|
|
|
|
BOOST_CHECK_EQUAL("2006", Utility::FormatDateTime("%Y", ts));
|
|
|
|
BOOST_CHECK_EQUAL("2006#2006", Utility::FormatDateTime("%Y#%Y", ts));
|
|
|
|
BOOST_CHECK_EQUAL("%", Utility::FormatDateTime("%%", ts));
|
|
|
|
BOOST_CHECK_EQUAL("%Y", Utility::FormatDateTime("%%Y", ts));
|
|
|
|
BOOST_CHECK_EQUAL("", Utility::FormatDateTime("", ts));
|
|
|
|
BOOST_CHECK_EQUAL("1970-01-01 00:00:00", Utility::FormatDateTime("%F %T", 0.0));
|
|
|
|
BOOST_CHECK_EQUAL("2038-01-19 03:14:07", Utility::FormatDateTime("%F %T", 2147483647.0)); // 2^31 - 1
|
|
|
|
if constexpr (sizeof(time_t) > sizeof(int32_t)) {
|
|
|
|
BOOST_CHECK_EQUAL("2100-03-14 13:37:42", Utility::FormatDateTime("%F %T", 4108714662.0)); // Past year 2038
|
|
|
|
} else {
|
|
|
|
BOOST_WARN_MESSAGE(false, "skipping test with past 2038 input due to 32 bit time_t");
|
|
|
|
}
|
|
|
|
|
|
|
|
// Negative (pre-1970) timestamps.
|
|
|
|
#ifdef _MSC_VER
|
|
|
|
// localtime_s() on Windows doesn't seem to like them and always errors out.
|
|
|
|
BOOST_CHECK_THROW(Utility::FormatDateTime("%F %T", -1.0), posix_error);
|
|
|
|
BOOST_CHECK_THROW(Utility::FormatDateTime("%F %T", -2147483648.0), posix_error); // -2^31
|
|
|
|
#else /* _MSC_VER */
|
|
|
|
BOOST_CHECK_EQUAL("1969-12-31 23:59:59", Utility::FormatDateTime("%F %T", -1.0));
|
|
|
|
BOOST_CHECK_EQUAL("1901-12-13 20:45:52", Utility::FormatDateTime("%F %T", -2147483648.0)); // -2^31
|
|
|
|
#endif /* _MSC_VER */
|
|
|
|
|
|
|
|
// Values right at the limits of time_t.
|
|
|
|
//
|
|
|
|
// With 64 bit time_t, there may not be an exact double representation of its min/max value, std::nextafter() is
|
|
|
|
// used to move the value towards 0 so that it's within the range of doubles that can be represented as time_t.
|
|
|
|
//
|
|
|
|
// These are expected to result in an error due to the intermediate struct tm not being able to represent these
|
|
|
|
// timestamps, so localtime_r() returns EOVERFLOW which makes the implementation throw an exception.
|
|
|
|
BOOST_CHECK_THROW(Utility::FormatDateTime("%Y", std::nextafter(time_t_limit::min(), 0)), posix_error);
|
|
|
|
BOOST_CHECK_THROW(Utility::FormatDateTime("%Y", std::nextafter(time_t_limit::max(), 0)), posix_error);
|
|
|
|
}
|
|
|
|
|
2019-02-22 16:58:26 +01:00
|
|
|
BOOST_AUTO_TEST_SUITE_END()
|