2017-12-19 19:40:30 +08:00
|
|
|
#include <map>
|
2017-11-23 23:36:54 +08:00
|
|
|
#include <thread>
|
|
|
|
#include <vector>
|
|
|
|
|
2018-10-04 21:23:07 +08:00
|
|
|
#include <gmock/gmock.h>
|
|
|
|
#include <gtest/gtest.h>
|
2017-11-23 23:36:54 +08:00
|
|
|
|
2018-11-07 01:15:55 +08:00
|
|
|
#include "storage/common/types/types.hpp"
|
2018-10-09 17:09:10 +08:00
|
|
|
#include "storage/single_node/concurrent_id_mapper.hpp"
|
2017-11-23 23:36:54 +08:00
|
|
|
|
2018-05-10 22:33:41 +08:00
|
|
|
using IdLabel = storage::Label;
|
2018-10-09 17:09:10 +08:00
|
|
|
using MapperLabel = storage::ConcurrentIdMapper<IdLabel>;
|
2017-12-04 21:05:59 +08:00
|
|
|
|
2017-11-23 23:36:54 +08:00
|
|
|
TEST(ConcurrentIdMapper, SameValueGivesSameId) {
|
2018-05-10 22:33:41 +08:00
|
|
|
MapperLabel mapper;
|
2017-12-19 19:40:30 +08:00
|
|
|
EXPECT_EQ(mapper.value_to_id("a"), mapper.value_to_id("a"));
|
2017-11-23 23:36:54 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(ConcurrentIdMapper, IdToValue) {
|
2018-05-10 22:33:41 +08:00
|
|
|
MapperLabel mapper;
|
2017-12-19 19:40:30 +08:00
|
|
|
std::string value = "a";
|
2017-12-04 21:05:59 +08:00
|
|
|
auto id = mapper.value_to_id(value);
|
|
|
|
EXPECT_EQ(value, mapper.id_to_value(id));
|
2017-11-23 23:36:54 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(ConcurrentIdMapper, TwoValuesTwoIds) {
|
2018-05-10 22:33:41 +08:00
|
|
|
MapperLabel mapper;
|
2017-12-19 19:40:30 +08:00
|
|
|
EXPECT_NE(mapper.value_to_id("a"), mapper.value_to_id("b"));
|
2017-11-23 23:36:54 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(ConcurrentIdMapper, SameIdReturnedMultipleThreads) {
|
2017-12-19 19:40:30 +08:00
|
|
|
const int thread_count = 20;
|
|
|
|
std::vector<std::string> values;
|
|
|
|
for (int i = 0; i < 50; ++i) values.emplace_back("value" + std::to_string(i));
|
|
|
|
|
|
|
|
// Perform the whole test a number of times since it's stochastic (we're
|
|
|
|
// trying to detect bad behavior in parallel execution).
|
|
|
|
for (int loop_ind = 0; loop_ind < 20; ++loop_ind) {
|
2018-05-10 22:33:41 +08:00
|
|
|
MapperLabel mapper;
|
|
|
|
std::vector<std::map<IdLabel, std::string>> mappings(thread_count);
|
2017-12-19 19:40:30 +08:00
|
|
|
std::vector<std::thread> threads;
|
|
|
|
for (int thread_ind = 0; thread_ind < thread_count; ++thread_ind) {
|
|
|
|
threads.emplace_back([&mapper, &mappings, &values, thread_ind] {
|
|
|
|
auto &mapping = mappings[thread_ind];
|
|
|
|
for (auto &value : values) {
|
|
|
|
mapping.emplace(mapper.value_to_id(value), value);
|
2017-11-23 23:36:54 +08:00
|
|
|
}
|
2017-12-19 19:40:30 +08:00
|
|
|
});
|
|
|
|
}
|
|
|
|
for (auto &thread : threads) thread.join();
|
|
|
|
EXPECT_EQ(mappings[0].size(), values.size());
|
|
|
|
for (auto &mapping : mappings) EXPECT_EQ(mapping, mappings[0]);
|
|
|
|
}
|
2017-11-23 23:36:54 +08:00
|
|
|
}
|
2018-05-10 22:33:41 +08:00
|
|
|
|
|
|
|
using IdProperty = storage::Property;
|
2018-10-09 17:09:10 +08:00
|
|
|
using MapperProperty = storage::ConcurrentIdMapper<IdProperty>;
|
2018-05-10 22:33:41 +08:00
|
|
|
|
|
|
|
TEST(ConcurrentIdMapper, PropertyLocation) {
|
|
|
|
// TODO(ipaljak): write unit tests for storage::Common and all
|
|
|
|
// derived classes (tests/unit/storage_types.cpp)
|
|
|
|
std::string prop_on_disk_name = "test_name1";
|
|
|
|
std::string prop_in_mem_name = "test_name2";
|
|
|
|
std::vector<std::string> props_on_disk = {prop_on_disk_name};
|
|
|
|
MapperProperty mapper(props_on_disk);
|
|
|
|
|
|
|
|
auto on_disk = mapper.value_to_id(prop_on_disk_name);
|
|
|
|
ASSERT_EQ(on_disk.Id(), 0);
|
|
|
|
ASSERT_EQ(on_disk.Location(), storage::Location::Disk);
|
|
|
|
|
|
|
|
auto in_mem = mapper.value_to_id(prop_in_mem_name);
|
|
|
|
ASSERT_EQ(in_mem.Id(), 1);
|
|
|
|
ASSERT_EQ(in_mem.Location(), storage::Location::Memory);
|
|
|
|
}
|