| /* |
| * Copyright 2004 The WebRTC Project Authors. All rights reserved. |
| * |
| * Use of this source code is governed by a BSD-style license |
| * that can be found in the LICENSE file in the root of the source |
| * tree. An additional intellectual property rights grant can be found |
| * in the file PATENTS. All contributing project authors may |
| * be found in the AUTHORS file in the root of the source tree. |
| */ |
| |
| #include "rtc_base/crypto_random.h" |
| |
| #include <string.h> |
| |
| #include <cstring> |
| #include <string> |
| #include <utility> |
| |
| #include "rtc_base/buffer.h" |
| #include "test/gmock.h" |
| #include "test/gtest.h" |
| |
| namespace rtc { |
| namespace { |
| |
| using ::testing::_; |
| using ::testing::DoAll; |
| using ::testing::Invoke; |
| using ::testing::IsEmpty; |
| using ::testing::Not; |
| using ::testing::Return; |
| using ::testing::WithArg; |
| using ::testing::WithArgs; |
| |
| TEST(RandomTest, TestCreateRandomId) { |
| CreateRandomId(); |
| } |
| |
| TEST(RandomTest, TestCreateRandomDouble) { |
| for (int i = 0; i < 100; ++i) { |
| double r = CreateRandomDouble(); |
| EXPECT_GE(r, 0.0); |
| EXPECT_LT(r, 1.0); |
| } |
| } |
| |
| TEST(RandomTest, TestCreateNonZeroRandomId) { |
| EXPECT_NE(0U, CreateRandomNonZeroId()); |
| } |
| |
| TEST(RandomTest, TestCreateRandomString) { |
| std::string random = CreateRandomString(256); |
| EXPECT_EQ(256U, random.size()); |
| std::string random2; |
| EXPECT_TRUE(CreateRandomString(256, &random2)); |
| EXPECT_NE(random, random2); |
| EXPECT_EQ(256U, random2.size()); |
| } |
| |
| TEST(RandomTest, TestCreateRandomData) { |
| static size_t kRandomDataLength = 32; |
| std::string random1; |
| std::string random2; |
| EXPECT_TRUE(CreateRandomData(kRandomDataLength, &random1)); |
| EXPECT_EQ(kRandomDataLength, random1.size()); |
| EXPECT_TRUE(CreateRandomData(kRandomDataLength, &random2)); |
| EXPECT_EQ(kRandomDataLength, random2.size()); |
| EXPECT_NE(0, memcmp(random1.data(), random2.data(), kRandomDataLength)); |
| } |
| |
| TEST(RandomTest, TestCreateRandomStringEvenlyDivideTable) { |
| static std::string kUnbiasedTable("01234567"); |
| std::string random; |
| EXPECT_TRUE(CreateRandomString(256, kUnbiasedTable, &random)); |
| EXPECT_EQ(256U, random.size()); |
| |
| static std::string kBiasedTable("0123456789"); |
| EXPECT_FALSE(CreateRandomString(256, kBiasedTable, &random)); |
| EXPECT_EQ(0U, random.size()); |
| } |
| |
| TEST(RandomTest, TestCreateRandomUuid) { |
| std::string random = CreateRandomUuid(); |
| EXPECT_EQ(36U, random.size()); |
| } |
| |
| TEST(RandomTest, TestCreateRandomForTest) { |
| // Make sure we get the output we expect. |
| SetRandomTestMode(true); |
| EXPECT_EQ(2154761789U, CreateRandomId()); |
| EXPECT_EQ("h0ISP4S5SJKH/9EY", CreateRandomString(16)); |
| EXPECT_EQ("41706e92-cdd3-46d9-a22d-8ff1737ffb11", CreateRandomUuid()); |
| static size_t kRandomDataLength = 32; |
| std::string random; |
| EXPECT_TRUE(CreateRandomData(kRandomDataLength, &random)); |
| EXPECT_EQ(kRandomDataLength, random.size()); |
| Buffer expected( |
| "\xbd\x52\x2a\x4b\x97\x93\x2f\x1c" |
| "\xc4\x72\xab\xa2\x88\x68\x3e\xcc" |
| "\xa3\x8d\xaf\x13\x3b\xbc\x83\xbb" |
| "\x16\xf1\xcf\x56\x0c\xf5\x4a\x8b", |
| kRandomDataLength); |
| EXPECT_EQ(0, memcmp(expected.data(), random.data(), kRandomDataLength)); |
| |
| // Reset and make sure we get the same output. |
| SetRandomTestMode(true); |
| EXPECT_EQ(2154761789U, CreateRandomId()); |
| EXPECT_EQ("h0ISP4S5SJKH/9EY", CreateRandomString(16)); |
| EXPECT_EQ("41706e92-cdd3-46d9-a22d-8ff1737ffb11", CreateRandomUuid()); |
| EXPECT_TRUE(CreateRandomData(kRandomDataLength, &random)); |
| EXPECT_EQ(kRandomDataLength, random.size()); |
| EXPECT_EQ(0, memcmp(expected.data(), random.data(), kRandomDataLength)); |
| |
| // Test different character sets. |
| SetRandomTestMode(true); |
| std::string str; |
| EXPECT_TRUE(CreateRandomString(16, "a", &str)); |
| EXPECT_EQ("aaaaaaaaaaaaaaaa", str); |
| EXPECT_TRUE(CreateRandomString(16, "abcd", &str)); |
| EXPECT_EQ("dbaaabdaccbcabbd", str); |
| |
| // Turn off test mode for other tests. |
| SetRandomTestMode(false); |
| } |
| |
| class MockRandomGenerator : public RandomGenerator { |
| public: |
| MOCK_METHOD(void, Die, ()); |
| ~MockRandomGenerator() override { Die(); } |
| |
| MOCK_METHOD(bool, Init, (const void* seed, size_t len), (override)); |
| MOCK_METHOD(bool, Generate, (void* buf, size_t len), (override)); |
| }; |
| |
| TEST(RandomTest, TestSetRandomGenerator) { |
| std::unique_ptr<MockRandomGenerator> will_move = |
| std::make_unique<MockRandomGenerator>(); |
| MockRandomGenerator* generator = will_move.get(); |
| SetRandomGenerator(std::move(will_move)); |
| |
| EXPECT_CALL(*generator, Init(_, sizeof(int))).WillOnce(Return(true)); |
| EXPECT_TRUE(InitRandom(5)); |
| |
| std::string seed = "seed"; |
| EXPECT_CALL(*generator, Init(seed.data(), seed.size())) |
| .WillOnce(Return(true)); |
| EXPECT_TRUE(InitRandom(seed.data(), seed.size())); |
| |
| uint32_t id = 4658; |
| EXPECT_CALL(*generator, Generate(_, sizeof(uint32_t))) |
| .WillOnce(DoAll(WithArg<0>(Invoke([&id](void* p) { |
| std::memcpy(p, &id, sizeof(uint32_t)); |
| })), |
| Return(true))); |
| EXPECT_EQ(CreateRandomId(), id); |
| |
| EXPECT_CALL(*generator, Generate) |
| .WillOnce(DoAll( |
| WithArgs<0, 1>([](void* p, size_t len) { std::memset(p, 0, len); }), |
| Return(true))); |
| EXPECT_THAT(CreateRandomUuid(), Not(IsEmpty())); |
| |
| // Set the default random generator, and expect that mock generator is |
| // not used beyond this point. |
| EXPECT_CALL(*generator, Die); |
| EXPECT_CALL(*generator, Generate).Times(0); |
| SetDefaultRandomGenerator(); |
| EXPECT_THAT(CreateRandomUuid(), Not(IsEmpty())); |
| } |
| |
| } // namespace |
| } // namespace rtc |