mirror of
https://github.com/deepseek-ai/3FS
synced 2025-06-26 18:16:45 +00:00
166 lines
5.9 KiB
C++
166 lines
5.9 KiB
C++
#include <algorithm>
|
|
#include <chrono>
|
|
#include <thread>
|
|
|
|
#include "client/storage/StorageClient.h"
|
|
#include "common/kv/mem/MemKVEngine.h"
|
|
#include "common/net/Client.h"
|
|
#include "common/serde/Serde.h"
|
|
#include "fbs/mgmtd/RoutingInfo.h"
|
|
#include "fbs/storage/Service.h"
|
|
#include "storage/service/StorageOperator.h"
|
|
#include "storage/service/StorageServer.h"
|
|
#include "storage/store/ChunkMetadata.h"
|
|
#include "tests/GtestHelpers.h"
|
|
#include "tests/client/ClientWithConfig.h"
|
|
#include "tests/client/ServerWithConfig.h"
|
|
#include "tests/lib/Helper.h"
|
|
#include "tests/lib/UnitTestFabric.h"
|
|
#include "tests/mgmtd/MgmtdTestHelper.h"
|
|
|
|
namespace hf3fs::storage {
|
|
namespace {
|
|
|
|
using namespace hf3fs::test;
|
|
class TestStorageForward : public UnitTestFabric, public ::testing::Test {
|
|
protected:
|
|
TestStorageForward()
|
|
: UnitTestFabric(SystemSetupConfig{
|
|
128_KB /*chunkSize*/,
|
|
1 /*numChains*/,
|
|
3 /*numReplicas*/,
|
|
3 /*numStorageNodes*/,
|
|
{folly::fs::temp_directory_path()} /*dataPaths*/,
|
|
hf3fs::Path() /*clientConfig*/,
|
|
hf3fs::Path() /*serverConfig*/,
|
|
{} /*storageEndpoints*/,
|
|
0 /*serviceLevel*/,
|
|
0 /*listenPort*/,
|
|
client::StorageClient::ImplementationType::RPC /*clientImplType*/,
|
|
kv::KVStore::Type::RocksDB /*metaStoreType*/,
|
|
true /*useFakeMgmtdClient*/,
|
|
}) {}
|
|
|
|
void SetUp() override {
|
|
net::IBDevice::Config ibConfig;
|
|
auto ibResult = net::IBManager::start(ibConfig);
|
|
ASSERT_OK(ibResult);
|
|
ASSERT_TRUE(setUpStorageSystem());
|
|
}
|
|
|
|
void TearDown() override { tearDownStorageSystem(); }
|
|
};
|
|
|
|
TEST_F(TestStorageForward, Write) {
|
|
// register a block of memory
|
|
std::vector<uint8_t> memoryBlock(setupConfig_.chunk_size(), 0xFF);
|
|
auto regRes = storageClient_->registerIOBuffer(&memoryBlock[0], memoryBlock.size());
|
|
ASSERT_OK(regRes);
|
|
|
|
// create write IO
|
|
|
|
auto ioBuffer = std::move(*regRes);
|
|
auto chainId = firstChainId_;
|
|
ChunkId chunkId(1 /*high*/, 1 /*low*/);
|
|
auto writeIO = storageClient_->createWriteIO(chainId,
|
|
chunkId,
|
|
0 /*offset*/,
|
|
setupConfig_.chunk_size() /*length*/,
|
|
setupConfig_.chunk_size() /*chunkSize*/,
|
|
&memoryBlock[0],
|
|
&ioBuffer);
|
|
|
|
client::WriteOptions options;
|
|
|
|
storageServers_.back()->stopAndJoin();
|
|
storageServers_.pop_back();
|
|
|
|
ASSERT_TRUE(updateRoutingInfo([&](auto &routingInfo) {
|
|
setTargetOffline(routingInfo, 0);
|
|
setTargetOffline(routingInfo, 1);
|
|
}));
|
|
|
|
folly::coro::blockingWait(storageClient_->write(writeIO, flat::UserInfo(), options));
|
|
ASSERT_OK(writeIO.result.lengthInfo);
|
|
ASSERT_EQ(writeIO.length, writeIO.result.lengthInfo.value());
|
|
}
|
|
|
|
TEST_F(TestStorageForward, WriteFailed) {
|
|
// register a block of memory
|
|
std::vector<uint8_t> memoryBlock(setupConfig_.chunk_size(), 0xFF);
|
|
auto regRes = storageClient_->registerIOBuffer(&memoryBlock[0], memoryBlock.size());
|
|
ASSERT_OK(regRes);
|
|
|
|
// create write IO
|
|
|
|
auto ioBuffer = std::move(*regRes);
|
|
auto chainId = firstChainId_;
|
|
ChunkId chunkId(1 /*high*/, 1 /*low*/);
|
|
auto writeIO = storageClient_->createWriteIO(chainId,
|
|
chunkId,
|
|
0 /*offset*/,
|
|
setupConfig_.chunk_size() /*length*/,
|
|
setupConfig_.chunk_size() /*chunkSize*/,
|
|
&memoryBlock[0],
|
|
&ioBuffer);
|
|
|
|
client::WriteOptions options;
|
|
|
|
storageServers_.back()->stopAndJoin();
|
|
storageServers_.pop_back();
|
|
|
|
clientConfig_.retry().set_max_retry_time(10_s);
|
|
folly::coro::blockingWait(storageClient_->write(writeIO, flat::UserInfo(), options));
|
|
ASSERT_FALSE(writeIO.result.lengthInfo);
|
|
}
|
|
|
|
TEST_F(TestStorageForward, WriteAndRead) {
|
|
// register a block of memory
|
|
std::vector<uint8_t> memoryBlock(setupConfig_.chunk_size(), 0xFF);
|
|
memoryBlock[0] = 0x01;
|
|
memoryBlock[1] = 0x02;
|
|
memoryBlock[2] = 0x03;
|
|
auto regRes = storageClient_->registerIOBuffer(&memoryBlock[0], memoryBlock.size());
|
|
ASSERT_OK(regRes);
|
|
|
|
// create write IO
|
|
|
|
auto ioBuffer = std::move(*regRes);
|
|
auto chainId = firstChainId_;
|
|
ChunkId chunkId(1 /*high*/, 1 /*low*/);
|
|
auto writeIO = storageClient_->createWriteIO(chainId,
|
|
chunkId,
|
|
0 /*offset*/,
|
|
3,
|
|
setupConfig_.chunk_size() /*chunkSize*/,
|
|
&memoryBlock[0],
|
|
&ioBuffer);
|
|
|
|
client::WriteOptions options;
|
|
folly::coro::blockingWait(storageClient_->write(writeIO, flat::UserInfo(), options));
|
|
ASSERT_OK(writeIO.result.lengthInfo);
|
|
ASSERT_EQ(writeIO.length, writeIO.result.lengthInfo.value());
|
|
|
|
{
|
|
auto readIO = storageClient_->createReadIO(chainId, chunkId, 0, 4096, &memoryBlock[0], &ioBuffer);
|
|
folly::coro::blockingWait(storageClient_->read(readIO, flat::UserInfo(), client::ReadOptions{}));
|
|
ASSERT_OK(readIO.result.lengthInfo);
|
|
ASSERT_EQ(*readIO.result.lengthInfo, 3);
|
|
ASSERT_EQ(memoryBlock[0], 0x01);
|
|
ASSERT_EQ(memoryBlock[1], 0x02);
|
|
ASSERT_EQ(memoryBlock[2], 0x03);
|
|
}
|
|
|
|
{
|
|
auto readIO = storageClient_->createReadIO(chainId, chunkId, 1, 4096, &memoryBlock[0], &ioBuffer);
|
|
folly::coro::blockingWait(storageClient_->read(readIO, flat::UserInfo(), client::ReadOptions{}));
|
|
ASSERT_OK(readIO.result.lengthInfo);
|
|
ASSERT_EQ(*readIO.result.lengthInfo, 2);
|
|
ASSERT_EQ(memoryBlock[0], 0x02);
|
|
ASSERT_EQ(memoryBlock[1], 0x03);
|
|
}
|
|
}
|
|
|
|
} // namespace
|
|
} // namespace hf3fs::storage
|