| // Copyright 2012 The Chromium Authors | 
 | // 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 <utility> | 
 |  | 
 | #include "testing/gtest/include/gtest/gtest.h" | 
 |  | 
 | namespace { | 
 |  | 
 | TEST(VersionTest, DefaultConstructor) { | 
 |   base::Version v; | 
 |   EXPECT_FALSE(v.IsValid()); | 
 | } | 
 |  | 
 | TEST(VersionTest, ValueSemantics) { | 
 |   base::Version v1("1.2.3.4"); | 
 |   EXPECT_TRUE(v1.IsValid()); | 
 |   base::Version v3; | 
 |   EXPECT_FALSE(v3.IsValid()); | 
 |   { | 
 |     base::Version v2(v1); | 
 |     v3 = v2; | 
 |     EXPECT_TRUE(v2.IsValid()); | 
 |     EXPECT_EQ(v1, v2); | 
 |   } | 
 |   EXPECT_EQ(v3, v1); | 
 | } | 
 |  | 
 | TEST(VersionTest, MoveSemantics) { | 
 |   const std::vector<uint32_t> components = {1, 2, 3, 4}; | 
 |   base::Version v1(std::move(components)); | 
 |   EXPECT_TRUE(v1.IsValid()); | 
 |   base::Version v2("1.2.3.4"); | 
 |   EXPECT_EQ(v1, v2); | 
 | } | 
 |  | 
 | 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 (const auto& i : cases) { | 
 |     base::Version version(i.input); | 
 |     EXPECT_EQ(i.success, version.IsValid()); | 
 |     if (i.success) { | 
 |       EXPECT_EQ(i.parts, version.components().size()); | 
 |       EXPECT_EQ(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}, | 
 |       {"15.5.28.130162", "15.5.28.130162", 0}, | 
 |   }; | 
 |   for (const auto& i : cases) { | 
 |     base::Version lhs(i.lhs); | 
 |     base::Version rhs(i.rhs); | 
 |     EXPECT_EQ(lhs.CompareTo(rhs), i.expected) << i.lhs << " ? " << i.rhs; | 
 |     // CompareToWildcardString() should have same behavior as CompareTo() when | 
 |     // no wildcards are present. | 
 |     EXPECT_EQ(lhs.CompareToWildcardString(i.rhs), i.expected) | 
 |         << i.lhs << " ? " << i.rhs; | 
 |     EXPECT_EQ(rhs.CompareToWildcardString(i.lhs), -i.expected) | 
 |         << i.lhs << " ? " << i.rhs; | 
 |  | 
 |     // Test comparison operators | 
 |     switch (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 (const auto& i : cases) { | 
 |     const base::Version version(i.lhs); | 
 |     const int result = version.CompareToWildcardString(i.rhs); | 
 |     EXPECT_EQ(result, i.expected) << i.lhs << "?" << 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 (const auto& i : cases) { | 
 |     EXPECT_EQ(base::Version::IsValidWildcardString(i.version), i.expected) | 
 |         << i.version << "?" << i.expected; | 
 |   } | 
 | } | 
 |  | 
 | TEST(VersionTest, LeadingZeros) { | 
 |   { | 
 |     // Leading zeros in the first component are not allowed. | 
 |     base::Version v("01.1"); | 
 |     EXPECT_FALSE(v.IsValid()); | 
 |   } | 
 |  | 
 |   { | 
 |     // Leading zeros in subsequent components are allowed (and this behavior is | 
 |     // now important for compatibility with existing modules, like extensions), | 
 |     // but are ignored because the value is parsed as an integer... | 
 |     base::Version v1("1.01"); | 
 |     EXPECT_TRUE(v1.IsValid()); | 
 |     // ...and as a result, v1.01 == v1.1. | 
 |     EXPECT_EQ("1.1", v1.GetString()); | 
 |     base::Version v2("1.1"); | 
 |     EXPECT_EQ(v1, v2); | 
 |   } | 
 |  | 
 |   // Similarly, since leading zeros are ignored, v1.02 > v1.1 (because | 
 |   // v1.02 is translated to 1.2). | 
 |   EXPECT_GT(base::Version("1.02"), base::Version("1.1")); | 
 | } | 
 |  | 
 | TEST(VersionTest, GetString) { | 
 |   static const struct version_compare { | 
 |     const char* version; | 
 |     bool valid; | 
 |     const char* string; | 
 |   } cases[] = { | 
 |       {"", false, "invalid"}, | 
 |       {"1", true, "1"}, | 
 |       {"1.0", true, "1.0"}, | 
 |       {"0.0.1.0", true, "0.0.1.0"}, | 
 |       {"1.2.3.4.5.6", true, "1.2.3.4.5.6"}, | 
 |       {"1.*.3", false, "invalid"}, | 
 |   }; | 
 |  | 
 |   for (const auto& i : cases) { | 
 |     base::Version v(i.version); | 
 |     EXPECT_EQ(v.IsValid(), i.valid); | 
 |     EXPECT_EQ(v.GetString(), i.string); | 
 |   } | 
 | } | 
 |  | 
 | }  // namespace |