2016-02-11 15:19:58 +01:00
|
|
|
#include "blockstore/interface/BlockStore.h"
|
2017-09-18 20:42:07 +01:00
|
|
|
#include <gtest/gtest.h>
|
|
|
|
#include <gmock/gmock.h>
|
|
|
|
#include <cpp-utils/data/DataFixture.h>
|
|
|
|
|
|
|
|
using ::testing::Test;
|
|
|
|
using ::testing::_;
|
|
|
|
using ::testing::Return;
|
|
|
|
using ::testing::Invoke;
|
|
|
|
using ::testing::Eq;
|
|
|
|
using ::testing::ByRef;
|
2019-10-13 23:49:57 +08:00
|
|
|
using ::testing::Action;
|
2017-09-18 20:42:07 +01:00
|
|
|
|
|
|
|
using std::string;
|
|
|
|
using cpputils::Data;
|
|
|
|
using cpputils::DataFixture;
|
|
|
|
using cpputils::unique_ref;
|
2019-10-13 23:49:57 +08:00
|
|
|
using cpputils::make_unique_ref;
|
2017-09-18 20:42:07 +01:00
|
|
|
using boost::optional;
|
|
|
|
|
|
|
|
using namespace blockstore;
|
|
|
|
|
|
|
|
class BlockStoreMock: public BlockStore {
|
|
|
|
public:
|
2019-10-13 23:49:57 +08:00
|
|
|
MOCK_METHOD(BlockId, createBlockId, (), (override));
|
|
|
|
MOCK_METHOD(optional<unique_ref<Block>>, tryCreate, (const BlockId &, Data data), (override));
|
|
|
|
MOCK_METHOD(unique_ref<Block>, overwrite, (const BlockId &, Data data), (override));
|
|
|
|
MOCK_METHOD(optional<unique_ref<Block>>, load, (const BlockId &), (override));
|
|
|
|
MOCK_METHOD(void, remove, (unique_ref<Block>), (override));
|
|
|
|
MOCK_METHOD(void, remove, (const BlockId &), (override));
|
|
|
|
MOCK_METHOD(uint64_t, numBlocks, (), (const, override));
|
|
|
|
MOCK_METHOD(uint64_t, estimateNumFreeBytes, (), (const, override));
|
|
|
|
MOCK_METHOD(uint64_t, blockSizeFromPhysicalBlockSize, (uint64_t), (const, override));
|
|
|
|
MOCK_METHOD(void, forEachBlock, (std::function<void (const blockstore::BlockId &)>), (const, override));
|
2017-09-18 20:42:07 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
class BlockMock: public Block {
|
|
|
|
public:
|
|
|
|
BlockMock(): Block(BlockId::Random()) {}
|
2019-10-13 23:49:57 +08:00
|
|
|
MOCK_METHOD(const void*, data, (), (const, override));
|
|
|
|
MOCK_METHOD(void, write, (const void*, uint64_t, uint64_t), (override));
|
|
|
|
MOCK_METHOD(void, flush, (), (override));
|
|
|
|
MOCK_METHOD(size_t, size, (), (const, override));
|
|
|
|
MOCK_METHOD(void, resize, (size_t), (override));
|
2017-09-18 20:42:07 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
class BlockStoreTest: public Test {
|
|
|
|
public:
|
|
|
|
BlockStoreTest() :blockStoreMock(), blockStore(blockStoreMock),
|
|
|
|
blockId1(BlockId::FromString("1491BB4932A389EE14BC7090AC772972")),
|
|
|
|
blockId2(BlockId::FromString("AC772971491BB4932A389EE14BC7090A")),
|
|
|
|
blockId3(BlockId::FromString("1BB4932A38AC77C7090A2971499EE14B")) {}
|
|
|
|
|
|
|
|
BlockStoreMock blockStoreMock;
|
|
|
|
BlockStore &blockStore;
|
|
|
|
const BlockId blockId1;
|
|
|
|
const BlockId blockId2;
|
|
|
|
const BlockId blockId3;
|
|
|
|
|
|
|
|
Data createDataWithSize(size_t size) {
|
|
|
|
Data fixture(DataFixture::generate(size));
|
|
|
|
Data data(size);
|
|
|
|
std::memcpy(data.data(), fixture.data(), size);
|
|
|
|
return data;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2019-10-13 23:49:57 +08:00
|
|
|
const Action<optional<unique_ref<Block>>(const BlockId &, cpputils::Data)> ReturnNewBlockMock = Invoke(
|
|
|
|
[] (const BlockId&, cpputils::Data) {
|
|
|
|
return optional<unique_ref<Block>>(unique_ref<Block>(make_unique_ref<BlockMock>()));
|
|
|
|
});
|
|
|
|
|
2017-09-18 20:42:07 +01:00
|
|
|
TEST_F(BlockStoreTest, DataIsPassedThrough0) {
|
|
|
|
Data data = createDataWithSize(0);
|
|
|
|
EXPECT_CALL(blockStoreMock, createBlockId()).WillOnce(Return(blockId1));
|
2019-10-13 23:49:57 +08:00
|
|
|
EXPECT_CALL(blockStoreMock, tryCreate(_, Eq(ByRef(data)))).WillOnce(ReturnNewBlockMock);
|
2017-09-18 20:42:07 +01:00
|
|
|
blockStore.create(data);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(BlockStoreTest, DataIsPassedThrough1) {
|
|
|
|
Data data = createDataWithSize(1);
|
|
|
|
EXPECT_CALL(blockStoreMock, createBlockId()).WillOnce(Return(blockId1));
|
2019-10-13 23:49:57 +08:00
|
|
|
EXPECT_CALL(blockStoreMock, tryCreate(_, Eq(ByRef(data)))).WillOnce(ReturnNewBlockMock);
|
2017-09-18 20:42:07 +01:00
|
|
|
blockStore.create(data);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(BlockStoreTest, DataIsPassedThrough1024) {
|
|
|
|
Data data = createDataWithSize(1024);
|
|
|
|
EXPECT_CALL(blockStoreMock, createBlockId()).WillOnce(Return(blockId1));
|
2019-10-13 23:49:57 +08:00
|
|
|
EXPECT_CALL(blockStoreMock, tryCreate(_, Eq(ByRef(data)))).WillOnce(ReturnNewBlockMock);
|
2017-09-18 20:42:07 +01:00
|
|
|
blockStore.create(data);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(BlockStoreTest, BlockIdIsCorrect) {
|
|
|
|
Data data = createDataWithSize(1024);
|
|
|
|
EXPECT_CALL(blockStoreMock, createBlockId()).WillOnce(Return(blockId1));
|
2019-10-13 23:49:57 +08:00
|
|
|
EXPECT_CALL(blockStoreMock, tryCreate(blockId1, _)).WillOnce(ReturnNewBlockMock);
|
2017-09-18 20:42:07 +01:00
|
|
|
blockStore.create(data);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(BlockStoreTest, TwoBlocksGetDifferentIds) {
|
|
|
|
EXPECT_CALL(blockStoreMock, createBlockId())
|
|
|
|
.WillOnce(Return(blockId1))
|
|
|
|
.WillOnce(Return(blockId2));
|
2019-10-13 23:49:57 +08:00
|
|
|
EXPECT_CALL(blockStoreMock, tryCreate(_, _))
|
|
|
|
.WillOnce(Invoke([this](const BlockId &blockId, Data) {
|
2017-09-18 20:42:07 +01:00
|
|
|
EXPECT_EQ(blockId1, blockId);
|
2019-10-13 23:49:57 +08:00
|
|
|
return optional<unique_ref<Block>>(unique_ref<Block>(make_unique_ref<BlockMock>()));
|
2017-09-18 20:42:07 +01:00
|
|
|
}))
|
2019-10-13 23:49:57 +08:00
|
|
|
.WillOnce(Invoke([this](const BlockId &blockId, Data) {
|
2017-09-18 20:42:07 +01:00
|
|
|
EXPECT_EQ(blockId2, blockId);
|
2019-10-13 23:49:57 +08:00
|
|
|
return optional<unique_ref<Block>>(unique_ref<Block>(make_unique_ref<BlockMock>()));
|
2017-09-18 20:42:07 +01:00
|
|
|
}));
|
|
|
|
|
|
|
|
Data data = createDataWithSize(1024);
|
|
|
|
blockStore.create(data);
|
|
|
|
blockStore.create(data);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(BlockStoreTest, WillTryADifferentIdIfKeyAlreadyExists) {
|
|
|
|
Data data = createDataWithSize(1024);
|
|
|
|
EXPECT_CALL(blockStoreMock, createBlockId())
|
|
|
|
.WillOnce(Return(blockId1))
|
|
|
|
.WillOnce(Return(blockId2));
|
2019-10-13 23:49:57 +08:00
|
|
|
EXPECT_CALL(blockStoreMock, tryCreate(_, Eq(ByRef(data))))
|
|
|
|
.WillOnce(Invoke([this](const BlockId &blockId, Data ) {
|
2017-09-18 20:42:07 +01:00
|
|
|
EXPECT_EQ(blockId1, blockId);
|
2019-10-13 23:49:57 +08:00
|
|
|
return boost::none;
|
2017-09-18 20:42:07 +01:00
|
|
|
}))
|
2019-10-13 23:49:57 +08:00
|
|
|
.WillOnce(Invoke([this](const BlockId &blockId, Data ) {
|
2017-09-18 20:42:07 +01:00
|
|
|
EXPECT_EQ(blockId2, blockId);
|
2019-10-13 23:49:57 +08:00
|
|
|
return optional<unique_ref<Block>>(unique_ref<Block>(make_unique_ref<BlockMock>()));
|
2017-09-18 20:42:07 +01:00
|
|
|
}));
|
|
|
|
|
|
|
|
blockStore.create(data);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(BlockStoreTest, WillTryADifferentIdIfIdAlreadyExistsTwoTimes) {
|
|
|
|
Data data = createDataWithSize(1024);
|
|
|
|
EXPECT_CALL(blockStoreMock, createBlockId())
|
|
|
|
.WillOnce(Return(blockId1))
|
|
|
|
.WillOnce(Return(blockId2))
|
|
|
|
.WillOnce(Return(blockId3));
|
2019-10-13 23:49:57 +08:00
|
|
|
EXPECT_CALL(blockStoreMock, tryCreate(_, Eq(ByRef(data))))
|
|
|
|
.WillOnce(Invoke([this](const BlockId &blockId, Data) {
|
2017-09-18 20:42:07 +01:00
|
|
|
EXPECT_EQ(blockId1, blockId);
|
2019-10-13 23:49:57 +08:00
|
|
|
return boost::none;
|
2017-09-18 20:42:07 +01:00
|
|
|
}))
|
2019-10-13 23:49:57 +08:00
|
|
|
.WillOnce(Invoke([this](const BlockId &blockId, Data) {
|
2017-09-18 20:42:07 +01:00
|
|
|
EXPECT_EQ(blockId2, blockId);
|
2019-10-13 23:49:57 +08:00
|
|
|
return boost::none;
|
2017-09-18 20:42:07 +01:00
|
|
|
}))
|
2019-10-13 23:49:57 +08:00
|
|
|
.WillOnce(Invoke([this](const BlockId &blockId, Data) {
|
2017-09-18 20:42:07 +01:00
|
|
|
EXPECT_EQ(blockId3, blockId);
|
2019-10-13 23:49:57 +08:00
|
|
|
return optional<unique_ref<Block>>(unique_ref<Block>(make_unique_ref<BlockMock>()));
|
2017-09-18 20:42:07 +01:00
|
|
|
}));
|
|
|
|
|
|
|
|
blockStore.create(data);
|
|
|
|
}
|