1*6777b538SAndroid Build Coastguard Worker // Copyright 2012 The Chromium Authors
2*6777b538SAndroid Build Coastguard Worker // Use of this source code is governed by a BSD-style license that can be
3*6777b538SAndroid Build Coastguard Worker // found in the LICENSE file.
4*6777b538SAndroid Build Coastguard Worker
5*6777b538SAndroid Build Coastguard Worker #include "base/uuid.h"
6*6777b538SAndroid Build Coastguard Worker
7*6777b538SAndroid Build Coastguard Worker #include <stdint.h>
8*6777b538SAndroid Build Coastguard Worker
9*6777b538SAndroid Build Coastguard Worker #include <limits>
10*6777b538SAndroid Build Coastguard Worker #include <set>
11*6777b538SAndroid Build Coastguard Worker #include <string_view>
12*6777b538SAndroid Build Coastguard Worker #include <unordered_set>
13*6777b538SAndroid Build Coastguard Worker
14*6777b538SAndroid Build Coastguard Worker #include "base/strings/string_util.h"
15*6777b538SAndroid Build Coastguard Worker #include "build/build_config.h"
16*6777b538SAndroid Build Coastguard Worker #include "testing/gtest/include/gtest/gtest.h"
17*6777b538SAndroid Build Coastguard Worker
18*6777b538SAndroid Build Coastguard Worker namespace base {
19*6777b538SAndroid Build Coastguard Worker
20*6777b538SAndroid Build Coastguard Worker namespace {
21*6777b538SAndroid Build Coastguard Worker
22*6777b538SAndroid Build Coastguard Worker // The format of Uuid version 4 must be xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx,
23*6777b538SAndroid Build Coastguard Worker // where y is one of [8, 9, a, b].
IsValidV4(const Uuid & guid)24*6777b538SAndroid Build Coastguard Worker bool IsValidV4(const Uuid& guid) {
25*6777b538SAndroid Build Coastguard Worker const std::string& lowercase = guid.AsLowercaseString();
26*6777b538SAndroid Build Coastguard Worker return guid.is_valid() && lowercase[14] == '4' &&
27*6777b538SAndroid Build Coastguard Worker (lowercase[19] == '8' || lowercase[19] == '9' ||
28*6777b538SAndroid Build Coastguard Worker lowercase[19] == 'a' || lowercase[19] == 'b');
29*6777b538SAndroid Build Coastguard Worker }
30*6777b538SAndroid Build Coastguard Worker
31*6777b538SAndroid Build Coastguard Worker } // namespace
32*6777b538SAndroid Build Coastguard Worker
TEST(UuidTest,UuidBasicUniqueness)33*6777b538SAndroid Build Coastguard Worker TEST(UuidTest, UuidBasicUniqueness) {
34*6777b538SAndroid Build Coastguard Worker constexpr int kIterations = 10;
35*6777b538SAndroid Build Coastguard Worker for (int i = 0; i < kIterations; ++i) {
36*6777b538SAndroid Build Coastguard Worker const Uuid guid1 = Uuid::GenerateRandomV4();
37*6777b538SAndroid Build Coastguard Worker const Uuid guid2 = Uuid::GenerateRandomV4();
38*6777b538SAndroid Build Coastguard Worker EXPECT_NE(guid1, guid2);
39*6777b538SAndroid Build Coastguard Worker EXPECT_TRUE(guid1.is_valid());
40*6777b538SAndroid Build Coastguard Worker EXPECT_TRUE(IsValidV4(guid1));
41*6777b538SAndroid Build Coastguard Worker EXPECT_TRUE(guid2.is_valid());
42*6777b538SAndroid Build Coastguard Worker EXPECT_TRUE(IsValidV4(guid2));
43*6777b538SAndroid Build Coastguard Worker }
44*6777b538SAndroid Build Coastguard Worker }
45*6777b538SAndroid Build Coastguard Worker
46*6777b538SAndroid Build Coastguard Worker namespace {
47*6777b538SAndroid Build Coastguard Worker
TestUuidValidity(std::string_view input,bool case_insensitive,bool strict)48*6777b538SAndroid Build Coastguard Worker void TestUuidValidity(std::string_view input,
49*6777b538SAndroid Build Coastguard Worker bool case_insensitive,
50*6777b538SAndroid Build Coastguard Worker bool strict) {
51*6777b538SAndroid Build Coastguard Worker SCOPED_TRACE(input);
52*6777b538SAndroid Build Coastguard Worker {
53*6777b538SAndroid Build Coastguard Worker const Uuid guid = Uuid::ParseCaseInsensitive(input);
54*6777b538SAndroid Build Coastguard Worker EXPECT_EQ(case_insensitive, guid.is_valid());
55*6777b538SAndroid Build Coastguard Worker }
56*6777b538SAndroid Build Coastguard Worker {
57*6777b538SAndroid Build Coastguard Worker const Uuid guid = Uuid::ParseLowercase(input);
58*6777b538SAndroid Build Coastguard Worker EXPECT_EQ(strict, guid.is_valid());
59*6777b538SAndroid Build Coastguard Worker }
60*6777b538SAndroid Build Coastguard Worker }
61*6777b538SAndroid Build Coastguard Worker
62*6777b538SAndroid Build Coastguard Worker } // namespace
63*6777b538SAndroid Build Coastguard Worker
TEST(UuidTest,Validity)64*6777b538SAndroid Build Coastguard Worker TEST(UuidTest, Validity) {
65*6777b538SAndroid Build Coastguard Worker // Empty Uuid is invalid.
66*6777b538SAndroid Build Coastguard Worker EXPECT_FALSE(Uuid().is_valid());
67*6777b538SAndroid Build Coastguard Worker
68*6777b538SAndroid Build Coastguard Worker enum Parsability { kDoesntParse, kParsesCaseInsensitiveOnly, kAlwaysParses };
69*6777b538SAndroid Build Coastguard Worker
70*6777b538SAndroid Build Coastguard Worker static constexpr struct {
71*6777b538SAndroid Build Coastguard Worker std::string_view input;
72*6777b538SAndroid Build Coastguard Worker Parsability parsability;
73*6777b538SAndroid Build Coastguard Worker } kUuidValidity[] = {
74*6777b538SAndroid Build Coastguard Worker {"invalid", kDoesntParse},
75*6777b538SAndroid Build Coastguard Worker {"0123456789ab-cdef-fedc-ba98-76543210", kDoesntParse},
76*6777b538SAndroid Build Coastguard Worker {"0123456789abcdeffedcba9876543210", kDoesntParse},
77*6777b538SAndroid Build Coastguard Worker {"01234567-89Zz-ZzZz-ZzZz-Zz9876543210", kDoesntParse},
78*6777b538SAndroid Build Coastguard Worker {"DEADBEEFDEADBEEFDEADBEEFDEADBEEF", kDoesntParse},
79*6777b538SAndroid Build Coastguard Worker {"deadbeefWdeadXbeefYdeadZbeefdeadbeef", kDoesntParse},
80*6777b538SAndroid Build Coastguard Worker {"XXXdeadbeefWdeadXbeefYdeadZbeefdeadbeefXXX", kDoesntParse},
81*6777b538SAndroid Build Coastguard Worker {"01234567-89aB-cDeF-fEdC-bA9876543210", kParsesCaseInsensitiveOnly},
82*6777b538SAndroid Build Coastguard Worker {"DEADBEEF-DEAD-BEEF-DEAD-BEEFDEADBEEF", kParsesCaseInsensitiveOnly},
83*6777b538SAndroid Build Coastguard Worker {"00000000-0000-0000-0000-000000000000", kAlwaysParses},
84*6777b538SAndroid Build Coastguard Worker {"deadbeef-dead-beef-dead-beefdeadbeef", kAlwaysParses},
85*6777b538SAndroid Build Coastguard Worker };
86*6777b538SAndroid Build Coastguard Worker
87*6777b538SAndroid Build Coastguard Worker for (const auto& validity : kUuidValidity) {
88*6777b538SAndroid Build Coastguard Worker const bool case_insensitive = validity.parsability != kDoesntParse;
89*6777b538SAndroid Build Coastguard Worker const bool strict = validity.parsability == kAlwaysParses;
90*6777b538SAndroid Build Coastguard Worker TestUuidValidity(validity.input, case_insensitive, strict);
91*6777b538SAndroid Build Coastguard Worker }
92*6777b538SAndroid Build Coastguard Worker }
93*6777b538SAndroid Build Coastguard Worker
TEST(UuidTest,EqualityAndRoundTrip)94*6777b538SAndroid Build Coastguard Worker TEST(UuidTest, EqualityAndRoundTrip) {
95*6777b538SAndroid Build Coastguard Worker static constexpr char kCanonicalStr[] =
96*6777b538SAndroid Build Coastguard Worker "deadbeef-dead-4eef-bead-beefdeadbeef";
97*6777b538SAndroid Build Coastguard Worker
98*6777b538SAndroid Build Coastguard Worker const Uuid from_lower =
99*6777b538SAndroid Build Coastguard Worker Uuid::ParseCaseInsensitive(ToLowerASCII(kCanonicalStr));
100*6777b538SAndroid Build Coastguard Worker EXPECT_EQ(kCanonicalStr, from_lower.AsLowercaseString());
101*6777b538SAndroid Build Coastguard Worker
102*6777b538SAndroid Build Coastguard Worker const Uuid from_upper =
103*6777b538SAndroid Build Coastguard Worker Uuid::ParseCaseInsensitive(ToUpperASCII(kCanonicalStr));
104*6777b538SAndroid Build Coastguard Worker EXPECT_EQ(kCanonicalStr, from_upper.AsLowercaseString());
105*6777b538SAndroid Build Coastguard Worker
106*6777b538SAndroid Build Coastguard Worker EXPECT_EQ(from_lower, from_upper);
107*6777b538SAndroid Build Coastguard Worker
108*6777b538SAndroid Build Coastguard Worker // Invalid Uuids are equal.
109*6777b538SAndroid Build Coastguard Worker EXPECT_EQ(Uuid(), Uuid());
110*6777b538SAndroid Build Coastguard Worker }
111*6777b538SAndroid Build Coastguard Worker
TEST(UuidTest,UnorderedSet)112*6777b538SAndroid Build Coastguard Worker TEST(UuidTest, UnorderedSet) {
113*6777b538SAndroid Build Coastguard Worker std::unordered_set<Uuid, UuidHash> guid_set;
114*6777b538SAndroid Build Coastguard Worker
115*6777b538SAndroid Build Coastguard Worker static constexpr char kUuid1[] = "01234567-89ab-cdef-fedc-ba9876543210";
116*6777b538SAndroid Build Coastguard Worker guid_set.insert(Uuid::ParseCaseInsensitive(ToLowerASCII(kUuid1)));
117*6777b538SAndroid Build Coastguard Worker EXPECT_EQ(1u, guid_set.size());
118*6777b538SAndroid Build Coastguard Worker guid_set.insert(Uuid::ParseCaseInsensitive(ToUpperASCII(kUuid1)));
119*6777b538SAndroid Build Coastguard Worker EXPECT_EQ(1u, guid_set.size());
120*6777b538SAndroid Build Coastguard Worker
121*6777b538SAndroid Build Coastguard Worker static constexpr char kUuid2[] = "deadbeef-dead-beef-dead-beefdeadbeef";
122*6777b538SAndroid Build Coastguard Worker guid_set.insert(Uuid::ParseCaseInsensitive(ToLowerASCII(kUuid2)));
123*6777b538SAndroid Build Coastguard Worker EXPECT_EQ(2u, guid_set.size());
124*6777b538SAndroid Build Coastguard Worker guid_set.insert(Uuid::ParseCaseInsensitive(ToUpperASCII(kUuid2)));
125*6777b538SAndroid Build Coastguard Worker EXPECT_EQ(2u, guid_set.size());
126*6777b538SAndroid Build Coastguard Worker }
127*6777b538SAndroid Build Coastguard Worker
TEST(UuidTest,Set)128*6777b538SAndroid Build Coastguard Worker TEST(UuidTest, Set) {
129*6777b538SAndroid Build Coastguard Worker std::set<Uuid> guid_set;
130*6777b538SAndroid Build Coastguard Worker
131*6777b538SAndroid Build Coastguard Worker static constexpr char kUuid1[] = "01234567-89ab-cdef-0123-456789abcdef";
132*6777b538SAndroid Build Coastguard Worker const Uuid guid1 = Uuid::ParseLowercase(kUuid1);
133*6777b538SAndroid Build Coastguard Worker ASSERT_TRUE(guid1.is_valid());
134*6777b538SAndroid Build Coastguard Worker guid_set.insert(guid1);
135*6777b538SAndroid Build Coastguard Worker
136*6777b538SAndroid Build Coastguard Worker static constexpr char kUuid2[] = "deadbeef-dead-beef-dead-beefdeadbeef";
137*6777b538SAndroid Build Coastguard Worker const Uuid guid2 = Uuid::ParseLowercase(kUuid2);
138*6777b538SAndroid Build Coastguard Worker ASSERT_TRUE(guid2.is_valid());
139*6777b538SAndroid Build Coastguard Worker guid_set.insert(guid2);
140*6777b538SAndroid Build Coastguard Worker
141*6777b538SAndroid Build Coastguard Worker // Test that the order of the Uuids was preserved.
142*6777b538SAndroid Build Coastguard Worker auto it = guid_set.begin();
143*6777b538SAndroid Build Coastguard Worker EXPECT_EQ(guid1, *it);
144*6777b538SAndroid Build Coastguard Worker ++it;
145*6777b538SAndroid Build Coastguard Worker EXPECT_EQ(guid2, *it);
146*6777b538SAndroid Build Coastguard Worker ++it;
147*6777b538SAndroid Build Coastguard Worker EXPECT_EQ(guid_set.end(), it);
148*6777b538SAndroid Build Coastguard Worker }
149*6777b538SAndroid Build Coastguard Worker
TEST(UuidTest,Compare)150*6777b538SAndroid Build Coastguard Worker TEST(UuidTest, Compare) {
151*6777b538SAndroid Build Coastguard Worker static constexpr char kUuid[] = "21abd97f-73e8-4b88-9389-a9fee6abda5e";
152*6777b538SAndroid Build Coastguard Worker static constexpr char kUuidLess[] = "1e0dcaca-9e7c-4f4b-bcc6-e4c02b0c99df";
153*6777b538SAndroid Build Coastguard Worker static constexpr char kUuidGreater[] = "6eeb1bc8-186b-433c-9d6a-a827bc96b2d4";
154*6777b538SAndroid Build Coastguard Worker
155*6777b538SAndroid Build Coastguard Worker const Uuid guid = Uuid::ParseLowercase(kUuid);
156*6777b538SAndroid Build Coastguard Worker const Uuid guid_eq = Uuid::ParseLowercase(kUuid);
157*6777b538SAndroid Build Coastguard Worker const Uuid guid_lt = Uuid::ParseLowercase(kUuidLess);
158*6777b538SAndroid Build Coastguard Worker const Uuid guid_gt = Uuid::ParseLowercase(kUuidGreater);
159*6777b538SAndroid Build Coastguard Worker const Uuid guid_invalid = Uuid();
160*6777b538SAndroid Build Coastguard Worker
161*6777b538SAndroid Build Coastguard Worker EXPECT_TRUE(guid_eq == guid);
162*6777b538SAndroid Build Coastguard Worker EXPECT_FALSE(guid_eq != guid);
163*6777b538SAndroid Build Coastguard Worker EXPECT_FALSE(guid_eq < guid);
164*6777b538SAndroid Build Coastguard Worker EXPECT_TRUE(guid_eq <= guid);
165*6777b538SAndroid Build Coastguard Worker EXPECT_FALSE(guid_eq > guid);
166*6777b538SAndroid Build Coastguard Worker EXPECT_TRUE(guid_eq >= guid);
167*6777b538SAndroid Build Coastguard Worker
168*6777b538SAndroid Build Coastguard Worker EXPECT_FALSE(guid_lt == guid);
169*6777b538SAndroid Build Coastguard Worker EXPECT_TRUE(guid_lt != guid);
170*6777b538SAndroid Build Coastguard Worker EXPECT_TRUE(guid_lt < guid);
171*6777b538SAndroid Build Coastguard Worker EXPECT_TRUE(guid_lt <= guid);
172*6777b538SAndroid Build Coastguard Worker EXPECT_FALSE(guid_lt > guid);
173*6777b538SAndroid Build Coastguard Worker EXPECT_FALSE(guid_lt >= guid);
174*6777b538SAndroid Build Coastguard Worker
175*6777b538SAndroid Build Coastguard Worker EXPECT_FALSE(guid_gt == guid);
176*6777b538SAndroid Build Coastguard Worker EXPECT_TRUE(guid_gt != guid);
177*6777b538SAndroid Build Coastguard Worker EXPECT_FALSE(guid_gt < guid);
178*6777b538SAndroid Build Coastguard Worker EXPECT_FALSE(guid_gt <= guid);
179*6777b538SAndroid Build Coastguard Worker EXPECT_TRUE(guid_gt > guid);
180*6777b538SAndroid Build Coastguard Worker EXPECT_TRUE(guid_gt >= guid);
181*6777b538SAndroid Build Coastguard Worker
182*6777b538SAndroid Build Coastguard Worker // Invalid Uuids are the "least".
183*6777b538SAndroid Build Coastguard Worker EXPECT_FALSE(guid_invalid == guid);
184*6777b538SAndroid Build Coastguard Worker EXPECT_TRUE(guid_invalid != guid);
185*6777b538SAndroid Build Coastguard Worker EXPECT_TRUE(guid_invalid < guid);
186*6777b538SAndroid Build Coastguard Worker EXPECT_TRUE(guid_invalid <= guid);
187*6777b538SAndroid Build Coastguard Worker EXPECT_FALSE(guid_invalid > guid);
188*6777b538SAndroid Build Coastguard Worker EXPECT_FALSE(guid_invalid >= guid);
189*6777b538SAndroid Build Coastguard Worker }
190*6777b538SAndroid Build Coastguard Worker
TEST(UuidTest,FormatRandomDataAsV4)191*6777b538SAndroid Build Coastguard Worker TEST(UuidTest, FormatRandomDataAsV4) {
192*6777b538SAndroid Build Coastguard Worker static constexpr uint64_t bytes1a[] = {0x0123456789abcdefull,
193*6777b538SAndroid Build Coastguard Worker 0x5a5a5a5aa5a5a5a5ull};
194*6777b538SAndroid Build Coastguard Worker static constexpr uint64_t bytes1b[] = {bytes1a[0], bytes1a[1]};
195*6777b538SAndroid Build Coastguard Worker static constexpr uint64_t bytes2[] = {0xfffffffffffffffdull,
196*6777b538SAndroid Build Coastguard Worker 0xfffffffffffffffeull};
197*6777b538SAndroid Build Coastguard Worker static constexpr uint64_t bytes3[] = {0xfffffffffffffffdull,
198*6777b538SAndroid Build Coastguard Worker 0xfffffffffffffffcull};
199*6777b538SAndroid Build Coastguard Worker
200*6777b538SAndroid Build Coastguard Worker const Uuid guid1a =
201*6777b538SAndroid Build Coastguard Worker Uuid::FormatRandomDataAsV4ForTesting(as_bytes(make_span(bytes1a)));
202*6777b538SAndroid Build Coastguard Worker const Uuid guid1b =
203*6777b538SAndroid Build Coastguard Worker Uuid::FormatRandomDataAsV4ForTesting(as_bytes(make_span(bytes1b)));
204*6777b538SAndroid Build Coastguard Worker const Uuid guid2 =
205*6777b538SAndroid Build Coastguard Worker Uuid::FormatRandomDataAsV4ForTesting(as_bytes(make_span(bytes2)));
206*6777b538SAndroid Build Coastguard Worker const Uuid guid3 =
207*6777b538SAndroid Build Coastguard Worker Uuid::FormatRandomDataAsV4ForTesting(as_bytes(make_span(bytes3)));
208*6777b538SAndroid Build Coastguard Worker
209*6777b538SAndroid Build Coastguard Worker EXPECT_TRUE(guid1a.is_valid());
210*6777b538SAndroid Build Coastguard Worker EXPECT_TRUE(guid1b.is_valid());
211*6777b538SAndroid Build Coastguard Worker EXPECT_TRUE(guid2.is_valid());
212*6777b538SAndroid Build Coastguard Worker EXPECT_TRUE(guid3.is_valid());
213*6777b538SAndroid Build Coastguard Worker
214*6777b538SAndroid Build Coastguard Worker // The same input should give the same Uuid.
215*6777b538SAndroid Build Coastguard Worker EXPECT_EQ(guid1a, guid1b);
216*6777b538SAndroid Build Coastguard Worker
217*6777b538SAndroid Build Coastguard Worker EXPECT_NE(guid1a, guid2);
218*6777b538SAndroid Build Coastguard Worker EXPECT_NE(guid1a, guid3);
219*6777b538SAndroid Build Coastguard Worker EXPECT_NE(guid2, guid3);
220*6777b538SAndroid Build Coastguard Worker }
221*6777b538SAndroid Build Coastguard Worker
222*6777b538SAndroid Build Coastguard Worker } // namespace base
223