// Copyright (c) 2012 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "base/version.h" #include <stddef.h> #include <stdint.h> #include "base/macros.h" #include "testing/gtest/include/gtest/gtest.h" namespace { TEST(VersionTest, DefaultConstructor) { Version v; EXPECT_FALSE(v.IsValid()); } TEST(VersionTest, ValueSemantics) { Version v1("1.2.3.4"); EXPECT_TRUE(v1.IsValid()); Version v3; EXPECT_FALSE(v3.IsValid()); { Version v2(v1); v3 = v2; EXPECT_TRUE(v2.IsValid()); EXPECT_EQ(v1, v2); } EXPECT_EQ(v3, v1); } TEST(VersionTest, GetVersionFromString) { static const struct version_string { const char* input; size_t parts; uint32_t firstpart; bool success; } cases[] = { {"", 0, 0, false}, {" ", 0, 0, false}, {"\t", 0, 0, false}, {"\n", 0, 0, false}, {" ", 0, 0, false}, {".", 0, 0, false}, {" . ", 0, 0, false}, {"0", 1, 0, true}, {"0.", 0, 0, false}, {"0.0", 2, 0, true}, {"4294967295.0", 2, 4294967295, true}, {"4294967296.0", 0, 0, false}, {"-1.0", 0, 0, false}, {"1.-1.0", 0, 0, false}, {"1,--1.0", 0, 0, false}, {"+1.0", 0, 0, false}, {"1.+1.0", 0, 0, false}, {"1+1.0", 0, 0, false}, {"++1.0", 0, 0, false}, {"1.0a", 0, 0, false}, {"1.2.3.4.5.6.7.8.9.0", 10, 1, true}, {"02.1", 0, 0, false}, {"0.01", 2, 0, true}, {"f.1", 0, 0, false}, {"15.007.20011", 3, 15, true}, {"15.5.28.130162", 4, 15, true}, }; for (size_t i = 0; i < arraysize(cases); ++i) { Version version(cases[i].input); EXPECT_EQ(cases[i].success, version.IsValid()); if (cases[i].success) { EXPECT_EQ(cases[i].parts, version.components().size()); EXPECT_EQ(cases[i].firstpart, version.components()[0]); } } } TEST(VersionTest, Compare) { static const struct version_compare { const char* lhs; const char* rhs; int expected; } cases[] = { {"1.0", "1.0", 0}, {"1.0", "0.0", 1}, {"1.0", "2.0", -1}, {"1.0", "1.1", -1}, {"1.1", "1.0", 1}, {"1.0", "1.0.1", -1}, {"1.1", "1.0.1", 1}, {"1.1", "1.0.1", 1}, {"1.0.0", "1.0", 0}, {"1.0.3", "1.0.20", -1}, {"11.0.10", "15.007.20011", -1}, {"11.0.10", "15.5.28.130162", -1}, }; for (size_t i = 0; i < arraysize(cases); ++i) { Version lhs(cases[i].lhs); Version rhs(cases[i].rhs); EXPECT_EQ(lhs.CompareTo(rhs), cases[i].expected) << cases[i].lhs << " ? " << cases[i].rhs; // Test comparison operators switch (cases[i].expected) { case -1: EXPECT_LT(lhs, rhs); EXPECT_LE(lhs, rhs); EXPECT_NE(lhs, rhs); EXPECT_FALSE(lhs == rhs); EXPECT_FALSE(lhs >= rhs); EXPECT_FALSE(lhs > rhs); break; case 0: EXPECT_FALSE(lhs < rhs); EXPECT_LE(lhs, rhs); EXPECT_FALSE(lhs != rhs); EXPECT_EQ(lhs, rhs); EXPECT_GE(lhs, rhs); EXPECT_FALSE(lhs > rhs); break; case 1: EXPECT_FALSE(lhs < rhs); EXPECT_FALSE(lhs <= rhs); EXPECT_NE(lhs, rhs); EXPECT_FALSE(lhs == rhs); EXPECT_GE(lhs, rhs); EXPECT_GT(lhs, rhs); break; } } } TEST(VersionTest, CompareToWildcardString) { static const struct version_compare { const char* lhs; const char* rhs; int expected; } cases[] = { {"1.0", "1.*", 0}, {"1.0", "0.*", 1}, {"1.0", "2.*", -1}, {"1.2.3", "1.2.3.*", 0}, {"10.0", "1.0.*", 1}, {"1.0", "3.0.*", -1}, {"1.4", "1.3.0.*", 1}, {"1.3.9", "1.3.*", 0}, {"1.4.1", "1.3.*", 1}, {"1.3", "1.4.5.*", -1}, {"1.5", "1.4.5.*", 1}, {"1.3.9", "1.3.*", 0}, {"1.2.0.0.0.0", "1.2.*", 0}, }; for (size_t i = 0; i < arraysize(cases); ++i) { const Version version(cases[i].lhs); const int result = version.CompareToWildcardString(cases[i].rhs); EXPECT_EQ(result, cases[i].expected) << cases[i].lhs << "?" << cases[i].rhs; } } TEST(VersionTest, IsValidWildcardString) { static const struct version_compare { const char* version; bool expected; } cases[] = { {"1.0", true}, {"", false}, {"1.2.3.4.5.6", true}, {"1.2.3.*", true}, {"1.2.3.5*", false}, {"1.2.3.56*", false}, {"1.*.3", false}, {"20.*", true}, {"+2.*", false}, {"*", false}, {"*.2", false}, }; for (size_t i = 0; i < arraysize(cases); ++i) { EXPECT_EQ(Version::IsValidWildcardString(cases[i].version), cases[i].expected) << cases[i].version << "?" << cases[i].expected; } } } // namespace