#define CATCH_CONFIG_MAIN #include "catch.hpp" #include "data_structures/map/rh_hashmultimap.hpp" class Data { private: size_t data = 0; int key; public: Data(int key) : key(key) {} const int &get_key() { return key; } }; void cross_validate(RhHashMultiMap &map, std::multimap &s_map); void cross_validate_weak(RhHashMultiMap &map, std::multimap &s_map); TEST_CASE("Robin hood hashmultimap basic functionality") { RhHashMultiMap map; REQUIRE(map.size() == 0); map.add(new Data(0)); REQUIRE(map.size() == 1); } TEST_CASE("Robin hood hashmultimap insert/get check") { RhHashMultiMap map; REQUIRE(map.find(0) == map.end()); auto ptr0 = new Data(0); map.add(ptr0); REQUIRE(map.find(0) != map.end()); REQUIRE(*map.find(0) == ptr0); } TEST_CASE("Robin hood hasmultihmap remove functionality") { RhHashMultiMap map; REQUIRE(map.find(0) == map.end()); auto ptr0 = new Data(0); map.add(ptr0); REQUIRE(map.find(0) != map.end()); REQUIRE(*map.find(0) == ptr0); REQUIRE(map.remove(0).get() == ptr0); REQUIRE(map.find(0) == map.end()); } TEST_CASE("Robin hood hashmultimap double insert") { RhHashMultiMap map; auto ptr0 = new Data(0); auto ptr1 = new Data(0); map.add(ptr0); map.add(ptr1); for (auto e : map) { if (ptr0 == e) { ptr0 = nullptr; continue; } if (ptr1 == e) { ptr1 = nullptr; continue; } REQUIRE(false); } } TEST_CASE("Robin hood hashmultimap") { RhHashMultiMap map; for (int i = 0; i < 128; i++) { REQUIRE(map.find(i) == map.end()); map.add(new Data(i)); REQUIRE(map.find(i) != map.end()); } for (int i = 0; i < 128; i++) { REQUIRE(map.find(i) != map.end()); REQUIRE(map.find(i)->get_key() == i); } } TEST_CASE("Robin hood hashmultimap iterate") { RhHashMultiMap map; for (int i = 0; i < 128; i++) { REQUIRE(map.find(i) == map.end()); map.add(new Data(i)); REQUIRE(map.find(i) != map.end()); } bool seen[128] = {false}; for (auto e : map) { auto key = e->get_key(); REQUIRE(!seen[key]); seen[key] = true; } for (int i = 0; i < 128; i++) { REQUIRE(seen[i]); } } TEST_CASE("Robin hood hashmultimap checked") { RhHashMultiMap map; std::multimap s_map; for (int i = 0; i < 1638; i++) { int key = (std::rand() % 100) << 3; auto data = new Data(key); map.add(data); s_map.insert(std::pair(key, data)); } cross_validate(map, s_map); } TEST_CASE("Robin hood hashmultimap checked rand") { RhHashMultiMap map; std::multimap s_map; std::srand(std::time(0)); for (int i = 0; i < 164308; i++) { int key = (std::rand() % 10000) << 3; auto data = new Data(key); map.add(data); s_map.insert(std::pair(key, data)); } cross_validate(map, s_map); } TEST_CASE("Robin hood hashmultimap with remove checked") { RhHashMultiMap map; std::multimap s_map; std::srand(std::time(0)); for (int i = 0; i < 162638; i++) { int key = (std::rand() % 10000) << 3; if ((std::rand() % 3) == 0) { auto removed = map.remove(key); auto it = s_map.find(key); if (removed.is_present()) { while (it != s_map.end() && it->second != removed.get()) { it++; } REQUIRE(it != s_map.end()); s_map.erase(it); } else { REQUIRE(it == s_map.end()); } } else { auto data = new Data(key); map.add(data); s_map.insert(std::pair(key, data)); } } cross_validate(map, s_map); } TEST_CASE("Robin hood hashmultimap with remove data checked") { RhHashMultiMap map; std::multimap s_map; std::srand(std::time(0)); for (int i = 0; i < 162638; i++) { int key = (std::rand() % 10000) << 3; if ((std::rand() % 2) == 0) { auto it = s_map.find(key); if (it == s_map.end()) { REQUIRE(map.find(key) == map.end()); } else { s_map.erase(it); REQUIRE(map.remove(it->second)); } } else { auto data = new Data(key); map.add(data); s_map.insert(std::pair(key, data)); } } cross_validate(map, s_map); } void cross_validate(RhHashMultiMap &map, std::multimap &s_map) { for (auto e : map) { auto it = s_map.find(e->get_key()); while (it != s_map.end() && it->second != e) { it++; } REQUIRE(it != s_map.end()); } for (auto e : s_map) { auto it = map.find(e.first); while (it != map.end() && *it != e.second) { it++; } REQUIRE(it != map.end()); } } void cross_validate_weak(RhHashMultiMap &map, std::multimap &s_map) { int count = 0; int key = 0; for (auto e : map) { if (e->get_key() == key) { count++; } else { auto it = s_map.find(key); while (it != s_map.end() && it->first == key) { it++; count--; } REQUIRE(count == 0); key = e->get_key(); count = 1; } } { auto it = s_map.find(key); while (it != s_map.end() && it->first == key) { it++; count--; } REQUIRE(count == 0); } for (auto e : s_map) { if (e.first == key) { count++; } else { auto it = map.find(key); while (it != map.end() && it->get_key() == key) { it++; count--; } REQUIRE(count == 0); key = e.first; count = 1; } } { auto it = map.find(key); while (it != map.end() && it->get_key() == key) { it++; count--; } REQUIRE(count == 0); } }