| // 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 |