blob: 122e9125c7c08f4ee18b6056fcc82b4d40381cac [file] [log] [blame]
// Copyright 2016 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 "components/url_pattern_index/closed_hash_map.h"
#include <string>
#include <vector>
#include "components/url_pattern_index/uint64_hasher.h"
#include "testing/gtest/include/gtest/gtest.h"
namespace url_pattern_index {
namespace {
template <typename MapType>
void ExpectHashMapIntegrity(const MapType& map, size_t min_capacity = 0) {
EXPECT_EQ(map.entries().size(), map.size());
EXPECT_EQ(map.hash_table().size(), map.table_size());
EXPECT_LE(map.size() * 2, map.table_size());
EXPECT_LE(min_capacity * 2, map.table_size());
std::vector<bool> entry_is_referenced(map.size());
for (size_t i = 0; i < map.table_size(); ++i) {
SCOPED_TRACE(testing::Message() << "Hash-table slot: " << i);
const uint32_t entry_index = map.hash_table()[i];
if (static_cast<size_t>(entry_index) >= map.size())
continue;
EXPECT_FALSE(entry_is_referenced[entry_index]);
entry_is_referenced[entry_index] = true;
}
for (size_t i = 0; i < map.size(); ++i) {
SCOPED_TRACE(testing::Message() << "Hash-table entry index: " << i);
EXPECT_TRUE(entry_is_referenced[i]);
}
}
template <typename MapType>
void ExpectEmptyMap(const MapType& map, size_t min_capacity) {
ExpectHashMapIntegrity(map, min_capacity);
EXPECT_EQ(0u, map.size());
}
} // namespace
TEST(ClosedHashMapTest, EmptyMapDefault) {
HashMap<int, int> hm;
ExpectEmptyMap(hm, 0);
EXPECT_EQ(nullptr, hm.Get(0));
EXPECT_EQ(nullptr, hm.Get(100500));
EXPECT_GT(hm.table_size(), 0u);
}
TEST(ClosedHashMapTest, EmptyMapWithCapacity) {
HashMap<int, int> hm(100);
ExpectEmptyMap(hm, 100);
EXPECT_EQ(nullptr, hm.Get(0));
EXPECT_EQ(nullptr, hm.Get(100500));
}
TEST(ClosedHashMapTest, InsertDistinctAndGet) {
HashMap<int, int> hm;
static const int kKeys[] = {1, 5, 10, 3, -100500};
for (int key : kKeys) {
EXPECT_TRUE(hm.Insert(key, -key));
ExpectHashMapIntegrity(hm);
}
for (int key : kKeys) {
const int* value = hm.Get(key);
ASSERT_NE(nullptr, value);
EXPECT_EQ(-key, *value);
}
EXPECT_EQ(nullptr, hm.Get(1234567));
}
TEST(ClosedHashMapTest, InsertExistingAndGet) {
HashMap<int, int> hm;
EXPECT_TRUE(hm.Insert(123, -123));
EXPECT_FALSE(hm.Insert(123, -124));
ExpectHashMapIntegrity(hm);
const int* value = hm.Get(123);
ASSERT_NE(nullptr, value);
EXPECT_EQ(-123, *value);
}
TEST(ClosedHashMapTest, InsertManyKeysWithCustomHasher) {
using CustomProber = SimpleQuadraticProber<uint64_t, Uint64Hasher>;
ClosedHashMap<uint64_t, std::string, CustomProber> hm;
ExpectEmptyMap(hm, 0);
std::vector<std::pair<uint64_t, std::string>> entries;
for (int key = 10, i = 0; key < 1000000; key += ++i) {
entries.push_back(std::make_pair(key, std::to_string(key)));
}
size_t expected_size = 0;
for (const auto& entry : entries) {
EXPECT_TRUE(hm.Insert(entry.first, entry.second));
EXPECT_FALSE(hm.Insert(entry.first, "-1"));
++expected_size;
EXPECT_EQ(expected_size, hm.size());
EXPECT_LE(expected_size * 2, hm.table_size());
}
ExpectHashMapIntegrity(hm);
for (const auto& entry : entries) {
const std::string* value = hm.Get(entry.first);
ASSERT_NE(nullptr, value);
EXPECT_EQ(entry.second, *value);
}
}
TEST(ClosedHashMapTest, OperatorBrackets) {
HashMap<int, int> hm;
for (int i = 0; i < 5; ++i) {
const size_t expected_size = i ? 1 : 0;
EXPECT_EQ(expected_size, hm.size());
int expected_value = (i + 1) * 10;
hm[123] = expected_value;
EXPECT_EQ(1u, hm.size());
const int* value_ptr = hm.Get(123);
ASSERT_NE(nullptr, value_ptr);
EXPECT_EQ(expected_value, *value_ptr);
EXPECT_EQ(expected_value, hm[123]);
EXPECT_EQ(1u, hm.size());
expected_value *= 100;
hm[123] = expected_value;
EXPECT_EQ(expected_value, hm[123]);
}
}
TEST(ClosedHashMapTest, ManualRehash) {
HashMap<int, int> hm(3);
const size_t expected_table_size = hm.table_size();
static const int kKeys[] = {1, 5, 10};
for (int key : kKeys) {
EXPECT_TRUE(hm.Insert(key, -key));
}
// No rehashing occurred.
EXPECT_EQ(expected_table_size, hm.table_size());
for (int i = 1; i <= 2; ++i) {
for (int key : kKeys) {
const int* value = hm.Get(key);
ASSERT_NE(nullptr, value);
EXPECT_EQ(-key, *value);
}
hm.Rehash(100 * i);
ExpectHashMapIntegrity(hm, 100 * i);
}
}
} // namespace url_pattern_index