Some checks failed
BlockStorage/repertory_osx/pipeline/head This commit looks good
BlockStorage/repertory_windows/pipeline/head This commit looks good
BlockStorage/repertory/pipeline/head There was a failure building this commit
BlockStorage/repertory_linux_builds/pipeline/head This commit looks good
BlockStorage/repertory_osx_builds/pipeline/head There was a failure building this commit
### Issues * \#1 \[bug\] Unable to mount S3 due to 'item_not_found' exception * \#2 Require bucket name for S3 mounts * \#3 \[bug\] File size is not being updated in S3 mount * \#4 Upgrade to libfuse-3.x.x * \#5 Switch to renterd for Sia support * \#6 Switch to cpp-httplib to further reduce dependencies * \#7 Remove global_data and calculate used disk space per provider * \#8 Switch to libcurl for S3 mount support ### Changes from v1.x.x * Added read-only encrypt provider * Pass-through mount point that transparently encrypts source data using `XChaCha20-Poly1305` * Added S3 encryption support via `XChaCha20-Poly1305` * Added replay protection to remote mounts * Added support base64 writes in remote FUSE * Created static linked Linux binaries for `amd64` and `aarch64` using `musl-libc` * Removed legacy Sia renter support * Removed Skynet support * Fixed multiple remote mount WinFSP API issues on \*NIX servers * Implemented chunked read and write * Writes for non-cached files are performed in chunks of 8Mib * Removed `repertory-ui` support * Removed `FreeBSD` support * Switched to `libsodium` over `CryptoPP` * Switched to `XChaCha20-Poly1305` for remote mounts * Updated `GoogleTest` to v1.14.0 * Updated `JSON for Modern C++` to v3.11.2 * Updated `OpenSSL` to v1.1.1w * Updated `RocksDB` to v8.5.3 * Updated `WinFSP` to 2023 * Updated `boost` to v1.78.0 * Updated `cURL` to v8.3.0 * Updated `zlib` to v1.3 * Use `upload_manager` for all providers * Adds a delay to uploads to prevent excessive API calls * Supports re-upload after mount restart for incomplete uploads * NOTE: Uploads for all providers are full file (no resume support) * Multipart upload support is planned for S3 Reviewed-on: #9
725 lines
26 KiB
C++
725 lines
26 KiB
C++
/*
|
|
Copyright <2018-2023> <scott.e.graves@protonmail.com>
|
|
|
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
of this software and associated documentation files (the "Software"), to deal
|
|
in the Software without restriction, including without limitation the rights
|
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
copies of the Software, and to permit persons to whom the Software is
|
|
furnished to do so, subject to the following conditions:
|
|
|
|
The above copyright notice and this permission notice shall be included in all
|
|
copies or substantial portions of the Software.
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
SOFTWARE.
|
|
*/
|
|
#if !_WIN32
|
|
|
|
#include "test_common.hpp"
|
|
|
|
#include "app_config.hpp"
|
|
#include "comm/curl/curl_comm.hpp"
|
|
#include "comm/s3/s3_comm.hpp"
|
|
#include "drives/fuse/fuse_drive.hpp"
|
|
#include "platform/platform.hpp"
|
|
#include "providers/s3/s3_provider.hpp"
|
|
#include "providers/sia/sia_provider.hpp"
|
|
#include "types/repertory.hpp"
|
|
#include "utils/event_capture.hpp"
|
|
#include "utils/file_utils.hpp"
|
|
#include "utils/utils.hpp"
|
|
|
|
#ifndef ACCESSPERMS
|
|
#define ACCESSPERMS (S_IRWXU | S_IRWXG | S_IRWXO) /* 0777 */
|
|
#endif
|
|
|
|
namespace repertory {
|
|
static void execute_mount(app_config &cfg, lock_data &ld, i_provider &provider,
|
|
const std::vector<std::string> &drive_args,
|
|
std::thread &th) {
|
|
EXPECT_EQ(0, fuse_drive(cfg, ld, provider).mount(drive_args));
|
|
th.join();
|
|
}
|
|
|
|
static void execute_unmount(i_provider &provider,
|
|
const std::string &mount_location) {
|
|
filesystem_item fsi{};
|
|
EXPECT_EQ(api_error::success, provider.get_filesystem_item("/", true, fsi));
|
|
EXPECT_STREQ("/", fsi.api_path.c_str());
|
|
EXPECT_TRUE(fsi.api_parent.empty());
|
|
EXPECT_TRUE(fsi.directory);
|
|
EXPECT_TRUE(fsi.encryption_token.empty());
|
|
EXPECT_EQ(std::uint64_t(0u), fsi.size);
|
|
EXPECT_TRUE(fsi.source_path.empty());
|
|
EXPECT_FALSE(fsi.is_encrypted());
|
|
|
|
api_meta_map meta{};
|
|
EXPECT_EQ(api_error::success, provider.get_item_meta("/", meta));
|
|
for (const auto &kv : meta) {
|
|
std::cout << kv.first << '=' << kv.second << std::endl;
|
|
}
|
|
|
|
auto unmounted = false;
|
|
for (int i = 0; not unmounted && (i < 50); i++) {
|
|
unmounted = (fuse_base::unmount(mount_location) == 0);
|
|
if (not unmounted) {
|
|
std::this_thread::sleep_for(100ms);
|
|
}
|
|
}
|
|
|
|
EXPECT_TRUE(unmounted);
|
|
}
|
|
|
|
static auto create_file_and_test(const std::string &mount_location,
|
|
std::string name) -> std::string {
|
|
auto file_path =
|
|
utils::path::absolute(utils::path::combine(mount_location, {name}));
|
|
|
|
auto fd =
|
|
open(file_path.c_str(), O_CREAT | O_RDWR, S_IRUSR | S_IWUSR | S_IRGRP);
|
|
EXPECT_LE(1, fd);
|
|
EXPECT_TRUE(utils::file::is_file(file_path));
|
|
EXPECT_FALSE(utils::file::is_directory(file_path));
|
|
|
|
std::uint64_t file_size{};
|
|
EXPECT_TRUE(utils::file::get_file_size(file_path, file_size));
|
|
EXPECT_EQ(0u, file_size);
|
|
|
|
EXPECT_EQ(0, close(fd));
|
|
std::this_thread::sleep_for(1s);
|
|
|
|
return file_path;
|
|
}
|
|
|
|
static void rmdir_and_test(const std::string &directory_path) {
|
|
int ret = 0;
|
|
for (auto i = 0; ((ret = rmdir(directory_path.c_str())) != 0) && (i < 20);
|
|
i++) {
|
|
std::this_thread::sleep_for(100ms);
|
|
}
|
|
|
|
EXPECT_EQ(0, ret);
|
|
std::this_thread::sleep_for(1s);
|
|
|
|
EXPECT_FALSE(utils::file::is_directory(directory_path));
|
|
EXPECT_FALSE(utils::file::is_file(directory_path));
|
|
}
|
|
|
|
static void unlink_file_and_test(const std::string &file_path) {
|
|
int ret = 0;
|
|
for (auto i = 0; ((ret = unlink(file_path.c_str())) != 0) && (i < 20); i++) {
|
|
std::this_thread::sleep_for(100ms);
|
|
}
|
|
|
|
EXPECT_EQ(0, ret);
|
|
|
|
std::this_thread::sleep_for(1s);
|
|
EXPECT_FALSE(utils::file::is_directory(file_path));
|
|
EXPECT_FALSE(utils::file::is_file(file_path));
|
|
}
|
|
|
|
static void test_chmod(const std::string &api_path,
|
|
const std::string &file_path, i_provider &provider) {
|
|
EXPECT_EQ(0, chmod(file_path.c_str(), S_IRUSR | S_IWUSR));
|
|
std::string mode;
|
|
EXPECT_EQ(api_error::success,
|
|
provider.get_item_meta(api_path.c_str(), META_MODE, mode));
|
|
EXPECT_EQ(static_cast<std::uint32_t>(S_IRUSR | S_IWUSR),
|
|
ACCESSPERMS & utils::string::to_uint32(mode));
|
|
}
|
|
|
|
static void test_chown(const std::string &api_path,
|
|
const std::string &file_path, i_provider &provider) {
|
|
EXPECT_EQ(0, chown(file_path.c_str(), -1, 0));
|
|
std::string gid;
|
|
EXPECT_EQ(api_error::success,
|
|
provider.get_item_meta(api_path, META_GID, gid));
|
|
EXPECT_STREQ("0", gid.c_str());
|
|
|
|
EXPECT_EQ(0, chown(file_path.c_str(), 0, -1));
|
|
std::string uid;
|
|
EXPECT_EQ(api_error::success,
|
|
provider.get_item_meta(api_path, META_UID, uid));
|
|
EXPECT_STREQ("0", uid.c_str());
|
|
}
|
|
|
|
static void test_mkdir(const std::string &api_path,
|
|
const std::string &directory_path,
|
|
i_provider &provider) {
|
|
EXPECT_EQ(0, mkdir(directory_path.c_str(),
|
|
S_IRUSR | S_IWUSR | S_IXUSR | S_IRGRP | S_IXGRP));
|
|
|
|
EXPECT_TRUE(utils::file::is_directory(directory_path));
|
|
EXPECT_FALSE(utils::file::is_file(directory_path));
|
|
|
|
std::string uid;
|
|
EXPECT_EQ(api_error::success,
|
|
provider.get_item_meta(api_path, META_UID, uid));
|
|
EXPECT_EQ(getuid(), utils::string::to_uint32(uid));
|
|
|
|
std::string gid;
|
|
EXPECT_EQ(api_error::success,
|
|
provider.get_item_meta(api_path, META_GID, gid));
|
|
EXPECT_EQ(getgid(), utils::string::to_uint32(gid));
|
|
|
|
std::string mode;
|
|
EXPECT_EQ(api_error::success,
|
|
provider.get_item_meta(api_path, META_MODE, mode));
|
|
EXPECT_EQ(static_cast<std::uint32_t>(S_IRUSR | S_IWUSR | S_IXUSR | S_IRGRP |
|
|
S_IXGRP),
|
|
ACCESSPERMS & (utils::string::to_uint32(mode)));
|
|
}
|
|
|
|
static void test_write_and_read(const std::string &api_path,
|
|
const std::string &file_path,
|
|
i_provider &provider) {
|
|
auto fd =
|
|
open(file_path.c_str(), O_CREAT | O_RDWR, S_IRUSR | S_IWUSR | S_IRGRP);
|
|
EXPECT_LE(1, fd);
|
|
|
|
std::string data = "TestData";
|
|
EXPECT_EQ(data.size(),
|
|
static_cast<std::size_t>(write(fd, &data[0], data.size())));
|
|
EXPECT_EQ(0, lseek(fd, 0, SEEK_SET));
|
|
|
|
data_buffer read_data;
|
|
read_data.resize(data.size());
|
|
EXPECT_EQ(data.size(), static_cast<std::size_t>(
|
|
read(fd, &read_data[0], read_data.size())));
|
|
|
|
EXPECT_EQ(0, memcmp(&data[0], &read_data[0], data.size()));
|
|
|
|
EXPECT_EQ(0, close(fd));
|
|
|
|
std::this_thread::sleep_for(1s);
|
|
|
|
std::uint64_t file_size{};
|
|
EXPECT_TRUE(utils::file::get_file_size(file_path, file_size));
|
|
EXPECT_EQ(data.size(), file_size);
|
|
|
|
filesystem_item fsi{};
|
|
EXPECT_EQ(api_error::success,
|
|
provider.get_filesystem_item(api_path, false, fsi));
|
|
EXPECT_TRUE(utils::file::get_file_size(fsi.source_path, file_size));
|
|
EXPECT_EQ(data.size(), file_size);
|
|
}
|
|
|
|
static void test_rename_file(const std::string &from_file_path,
|
|
const std::string &to_file_path,
|
|
i_provider &provider) {
|
|
auto fd = open(from_file_path.c_str(), O_RDWR, S_IRUSR | S_IWUSR | S_IRGRP);
|
|
EXPECT_LE(1, fd);
|
|
close(fd);
|
|
|
|
std::this_thread::sleep_for(1s);
|
|
|
|
if (provider.is_rename_supported()) {
|
|
EXPECT_EQ(0, rename(from_file_path.c_str(), to_file_path.c_str()));
|
|
EXPECT_FALSE(utils::file::is_file(from_file_path));
|
|
EXPECT_TRUE(utils::file::is_file(to_file_path));
|
|
} else {
|
|
EXPECT_EQ(-1, rename(from_file_path.c_str(), to_file_path.c_str()));
|
|
EXPECT_TRUE(utils::file::is_file(from_file_path));
|
|
EXPECT_FALSE(utils::file::is_file(to_file_path));
|
|
}
|
|
}
|
|
|
|
static void test_rename_directory(const std::string &from_dir_path,
|
|
const std::string &to_dir_path,
|
|
i_provider &provider) {
|
|
EXPECT_EQ(0, mkdir(from_dir_path.c_str(),
|
|
S_IRUSR | S_IWUSR | S_IXUSR | S_IRGRP | S_IXGRP));
|
|
std::this_thread::sleep_for(1s);
|
|
|
|
EXPECT_TRUE(utils::file::is_directory(from_dir_path));
|
|
if (provider.is_rename_supported()) {
|
|
EXPECT_EQ(0, rename(from_dir_path.c_str(), to_dir_path.c_str()));
|
|
EXPECT_FALSE(utils::file::is_directory(from_dir_path));
|
|
EXPECT_TRUE(utils::file::is_directory(to_dir_path));
|
|
} else {
|
|
EXPECT_EQ(-1, rename(from_dir_path.c_str(), to_dir_path.c_str()));
|
|
EXPECT_TRUE(utils::file::is_directory(from_dir_path));
|
|
EXPECT_FALSE(utils::file::is_directory(to_dir_path));
|
|
}
|
|
}
|
|
|
|
static void test_truncate(const std::string &file_path) {
|
|
EXPECT_EQ(0, truncate(file_path.c_str(), 10u));
|
|
|
|
std::uint64_t file_size{};
|
|
EXPECT_TRUE(utils::file::get_file_size(file_path, file_size));
|
|
|
|
EXPECT_EQ(std::uint64_t(10u), file_size);
|
|
}
|
|
|
|
static void test_ftruncate(const std::string &file_path) {
|
|
auto fd = open(file_path.c_str(), O_RDWR, S_IRUSR | S_IWUSR | S_IRGRP);
|
|
EXPECT_LE(1, fd);
|
|
|
|
EXPECT_EQ(0, ftruncate(fd, 10u));
|
|
|
|
std::uint64_t file_size{};
|
|
EXPECT_TRUE(utils::file::get_file_size(file_path, file_size));
|
|
|
|
EXPECT_EQ(std::uint64_t(10u), file_size);
|
|
|
|
close(fd);
|
|
}
|
|
|
|
#ifndef __APPLE__
|
|
static void test_fallocate(const std::string &api_path,
|
|
const std::string &file_path, i_provider &provider) {
|
|
auto fd =
|
|
open(file_path.c_str(), O_CREAT | O_RDWR, S_IRUSR | S_IWUSR | S_IRGRP);
|
|
EXPECT_LE(1, fd);
|
|
EXPECT_EQ(0, fallocate(fd, 0, 0, 16));
|
|
|
|
std::uint64_t file_size{};
|
|
EXPECT_TRUE(utils::file::get_file_size(file_path, file_size));
|
|
EXPECT_EQ(16u, file_size);
|
|
|
|
EXPECT_EQ(0, close(fd));
|
|
|
|
filesystem_item fsi{};
|
|
EXPECT_EQ(api_error::success,
|
|
provider.get_filesystem_item(api_path, false, fsi));
|
|
|
|
file_size = 0u;
|
|
EXPECT_TRUE(utils::file::get_file_size(fsi.source_path, file_size));
|
|
EXPECT_EQ(16u, file_size);
|
|
}
|
|
#endif
|
|
|
|
static void test_file_getattr(const std::string & /* api_path */,
|
|
const std::string &file_path,
|
|
i_provider & /* provider */) {
|
|
auto fd =
|
|
open(file_path.c_str(), O_CREAT | O_RDONLY, S_IRUSR | S_IWUSR | S_IRGRP);
|
|
EXPECT_LE(1, fd);
|
|
|
|
EXPECT_EQ(0, close(fd));
|
|
|
|
struct stat64 st {};
|
|
EXPECT_EQ(0, stat64(file_path.c_str(), &st));
|
|
EXPECT_EQ(static_cast<std::uint32_t>(S_IRUSR | S_IWUSR | S_IRGRP),
|
|
ACCESSPERMS & st.st_mode);
|
|
EXPECT_FALSE(S_ISDIR(st.st_mode));
|
|
EXPECT_TRUE(S_ISREG(st.st_mode));
|
|
}
|
|
|
|
static void test_directory_getattr(const std::string & /* api_path */,
|
|
const std::string &directory_path,
|
|
i_provider & /* provider */) {
|
|
EXPECT_EQ(0, mkdir(directory_path.c_str(),
|
|
S_IRUSR | S_IWUSR | S_IXUSR | S_IRGRP | S_IXGRP));
|
|
|
|
struct stat64 st {};
|
|
EXPECT_EQ(0, stat64(directory_path.c_str(), &st));
|
|
EXPECT_EQ(static_cast<std::uint32_t>(S_IRUSR | S_IWUSR | S_IXUSR | S_IRGRP |
|
|
S_IXGRP),
|
|
ACCESSPERMS & st.st_mode);
|
|
EXPECT_TRUE(S_ISDIR(st.st_mode));
|
|
EXPECT_FALSE(S_ISREG(st.st_mode));
|
|
}
|
|
|
|
static void
|
|
test_write_operations_fail_if_read_only(const std::string &api_path,
|
|
const std::string &file_path,
|
|
i_provider &provider) {
|
|
auto fd =
|
|
open(file_path.c_str(), O_CREAT | O_RDONLY, S_IRUSR | S_IWUSR | S_IRGRP);
|
|
EXPECT_LE(1, fd);
|
|
|
|
std::string data = "TestData";
|
|
EXPECT_EQ(-1, write(fd, &data[0], data.size()));
|
|
|
|
EXPECT_EQ(-1, ftruncate(fd, 9u));
|
|
|
|
#ifndef __APPLE__
|
|
EXPECT_EQ(-1, fallocate(fd, 0, 0, 16));
|
|
#endif
|
|
|
|
EXPECT_EQ(0, close(fd));
|
|
|
|
std::this_thread::sleep_for(1s);
|
|
|
|
std::uint64_t file_size{};
|
|
EXPECT_TRUE(utils::file::get_file_size(file_path, file_size));
|
|
EXPECT_EQ(std::size_t(0u), file_size);
|
|
|
|
filesystem_item fsi{};
|
|
EXPECT_EQ(api_error::success,
|
|
provider.get_filesystem_item(api_path, false, fsi));
|
|
EXPECT_TRUE(utils::file::get_file_size(fsi.source_path, file_size));
|
|
EXPECT_EQ(std::size_t(0u), file_size);
|
|
}
|
|
|
|
#if !__APPLE__ && HAS_SETXATTR
|
|
static void test_xattr_invalid_parameters(const std::string &file_path) {
|
|
std::string attr = "moose";
|
|
EXPECT_EQ(-1, setxattr(nullptr, "user.test_attr", &attr[0], attr.size(),
|
|
XATTR_CREATE));
|
|
EXPECT_EQ(errno, EFAULT);
|
|
EXPECT_EQ(-1, setxattr(file_path.c_str(), nullptr, &attr[0], attr.size(),
|
|
XATTR_CREATE));
|
|
EXPECT_EQ(errno, EFAULT);
|
|
EXPECT_EQ(-1, setxattr(file_path.c_str(), "user.test_attr", nullptr,
|
|
attr.size(), XATTR_CREATE));
|
|
EXPECT_EQ(errno, EFAULT);
|
|
EXPECT_EQ(0, setxattr(file_path.c_str(), "user.test_attr", nullptr, 0,
|
|
XATTR_CREATE));
|
|
}
|
|
|
|
static void test_xattr_create_and_get(const std::string &file_path) {
|
|
std::string attr = "moose";
|
|
EXPECT_EQ(0, setxattr(file_path.c_str(), "user.test_attr", &attr[0],
|
|
attr.size(), XATTR_CREATE));
|
|
|
|
std::string val;
|
|
val.resize(attr.size());
|
|
EXPECT_EQ(attr.size(),
|
|
static_cast<std::size_t>(getxattr(
|
|
file_path.c_str(), "user.test_attr", &val[0], val.size())));
|
|
EXPECT_STREQ(attr.c_str(), val.c_str());
|
|
}
|
|
|
|
static void test_xattr_listxattr(const std::string &file_path) {
|
|
std::string attr = "moose";
|
|
EXPECT_EQ(0, setxattr(file_path.c_str(), "user.test_attr", &attr[0],
|
|
attr.size(), XATTR_CREATE));
|
|
EXPECT_EQ(0, setxattr(file_path.c_str(), "user.test_attr2", &attr[0],
|
|
attr.size(), XATTR_CREATE));
|
|
|
|
std::string val;
|
|
val.resize(attr.size());
|
|
EXPECT_EQ(attr.size(),
|
|
static_cast<std::size_t>(getxattr(
|
|
file_path.c_str(), "user.test_attr", &val[0], val.size())));
|
|
EXPECT_STREQ(attr.c_str(), val.c_str());
|
|
|
|
std::string data;
|
|
auto size = listxattr(file_path.c_str(), &data[0u], 0u);
|
|
EXPECT_EQ(31, size);
|
|
|
|
data.resize(size);
|
|
EXPECT_EQ(size, listxattr(file_path.c_str(), &data[0u], size));
|
|
|
|
char *ptr = &data[0u];
|
|
EXPECT_STREQ("user.test_attr", ptr);
|
|
|
|
ptr += strlen(ptr) + 1;
|
|
EXPECT_STREQ("user.test_attr2", ptr);
|
|
}
|
|
|
|
static void test_xattr_replace(const std::string &file_path) {
|
|
std::string attr = "moose";
|
|
EXPECT_EQ(0, setxattr(file_path.c_str(), "user.test_attr", &attr[0],
|
|
attr.size(), XATTR_CREATE));
|
|
|
|
attr = "cow";
|
|
EXPECT_EQ(0, setxattr(file_path.c_str(), "user.test_attr", &attr[0],
|
|
attr.size(), XATTR_REPLACE));
|
|
|
|
std::string val;
|
|
val.resize(attr.size());
|
|
EXPECT_EQ(attr.size(),
|
|
static_cast<std::size_t>(getxattr(
|
|
file_path.c_str(), "user.test_attr", &val[0], val.size())));
|
|
EXPECT_STREQ(attr.c_str(), val.c_str());
|
|
}
|
|
|
|
static void test_xattr_default_create(const std::string &file_path) {
|
|
std::string attr = "moose";
|
|
EXPECT_EQ(0, setxattr(file_path.c_str(), "user.test_attr", &attr[0],
|
|
attr.size(), 0));
|
|
|
|
std::string val;
|
|
val.resize(attr.size());
|
|
EXPECT_EQ(attr.size(),
|
|
static_cast<std::size_t>(getxattr(
|
|
file_path.c_str(), "user.test_attr", &val[0], val.size())));
|
|
EXPECT_STREQ(attr.c_str(), val.c_str());
|
|
}
|
|
|
|
static void test_xattr_default_replace(const std::string &file_path) {
|
|
std::string attr = "moose";
|
|
EXPECT_EQ(0, setxattr(file_path.c_str(), "user.test_attr", &attr[0],
|
|
attr.size(), 0));
|
|
|
|
attr = "cow";
|
|
EXPECT_EQ(0, setxattr(file_path.c_str(), "user.test_attr", &attr[0],
|
|
attr.size(), 0));
|
|
|
|
std::string val;
|
|
val.resize(attr.size());
|
|
EXPECT_EQ(attr.size(),
|
|
static_cast<std::size_t>(getxattr(
|
|
file_path.c_str(), "user.test_attr", &val[0], val.size())));
|
|
EXPECT_STREQ(attr.c_str(), val.c_str());
|
|
}
|
|
|
|
static void test_xattr_create_fails_if_exists(const std::string &file_path) {
|
|
std::string attr = "moose";
|
|
EXPECT_EQ(0, setxattr(file_path.c_str(), "user.test_attr", &attr[0],
|
|
attr.size(), 0));
|
|
EXPECT_EQ(-1, setxattr(file_path.c_str(), "user.test_attr", &attr[0],
|
|
attr.size(), XATTR_CREATE));
|
|
EXPECT_EQ(EEXIST, errno);
|
|
}
|
|
|
|
static void
|
|
test_xattr_create_fails_if_not_exists(const std::string &file_path) {
|
|
std::string attr = "moose";
|
|
EXPECT_EQ(-1, setxattr(file_path.c_str(), "user.test_attr", &attr[0],
|
|
attr.size(), XATTR_REPLACE));
|
|
EXPECT_EQ(ENODATA, errno);
|
|
}
|
|
|
|
static void test_xattr_removexattr(const std::string &file_path) {
|
|
std::string attr = "moose";
|
|
EXPECT_EQ(0, setxattr(file_path.c_str(), "user.test_attr", &attr[0],
|
|
attr.size(), XATTR_CREATE));
|
|
|
|
EXPECT_EQ(0, removexattr(file_path.c_str(), "user.test_attr"));
|
|
|
|
std::string val;
|
|
val.resize(attr.size());
|
|
EXPECT_EQ(-1,
|
|
getxattr(file_path.c_str(), "user.test_attr", &val[0], val.size()));
|
|
EXPECT_EQ(ENODATA, errno);
|
|
}
|
|
#endif
|
|
|
|
TEST(fuse_drive, all_tests) {
|
|
auto current_directory = std::filesystem::current_path();
|
|
|
|
for (std::size_t idx = 0U; idx < 2U; idx++) {
|
|
std::filesystem::current_path(current_directory);
|
|
|
|
const auto test_directory =
|
|
utils::path::absolute("./fuse_drive" + std::to_string(idx));
|
|
|
|
const auto mount_location = utils::path::absolute(
|
|
utils::path::combine(test_directory, {"mount", std::to_string(idx)}));
|
|
EXPECT_TRUE(utils::file::delete_directory_recursively(mount_location));
|
|
EXPECT_TRUE(utils::file::create_full_directory_path(mount_location));
|
|
{
|
|
std::unique_ptr<app_config> config_ptr{};
|
|
#ifdef REPERTORY_ENABLE_S3
|
|
std::unique_ptr<s3_comm> s3_comm_ptr{};
|
|
#endif
|
|
std::unique_ptr<curl_comm> comm_ptr{};
|
|
std::unique_ptr<i_provider> provider_ptr{};
|
|
std::unique_ptr<lock_data> lock_data_ptr{};
|
|
|
|
const auto cfg_directory = utils::path::absolute(
|
|
utils::path::combine(test_directory, {"cfg", std::to_string(idx)}));
|
|
EXPECT_TRUE(utils::file::delete_directory_recursively(cfg_directory));
|
|
EXPECT_TRUE(utils::file::create_full_directory_path(cfg_directory));
|
|
|
|
switch (idx) {
|
|
case 0U: {
|
|
#ifdef REPERTORY_ENABLE_S3
|
|
config_ptr =
|
|
std::make_unique<app_config>(provider_type::s3, cfg_directory);
|
|
{
|
|
app_config src_cfg(
|
|
provider_type::s3,
|
|
utils::path::combine(get_test_dir(), {"filebase"}));
|
|
config_ptr->set_enable_drive_events(true);
|
|
config_ptr->set_event_level(event_level::verbose);
|
|
config_ptr->set_s3_config(src_cfg.get_s3_config());
|
|
}
|
|
|
|
s3_comm_ptr = std::make_unique<s3_comm>(*config_ptr);
|
|
provider_ptr = std::make_unique<s3_provider>(*config_ptr, *s3_comm_ptr);
|
|
lock_data_ptr = std::make_unique<lock_data>(
|
|
provider_type::s3, "unittests_" + std::to_string(idx));
|
|
#else
|
|
continue;
|
|
#endif
|
|
} break;
|
|
|
|
case 1U: {
|
|
config_ptr =
|
|
std::make_unique<app_config>(provider_type::sia, cfg_directory);
|
|
{
|
|
app_config src_cfg(provider_type::sia,
|
|
utils::path::combine(get_test_dir(), {"sia"}));
|
|
config_ptr->set_enable_drive_events(true);
|
|
config_ptr->set_event_level(event_level::debug);
|
|
config_ptr->set_host_config(src_cfg.get_host_config());
|
|
}
|
|
|
|
comm_ptr = std::make_unique<curl_comm>(config_ptr->get_host_config());
|
|
provider_ptr = std::make_unique<sia_provider>(*config_ptr, *comm_ptr);
|
|
lock_data_ptr = std::make_unique<lock_data>(
|
|
provider_type::sia, "unittests_" + std::to_string(idx));
|
|
} break;
|
|
|
|
default:
|
|
std::cerr << "provider at index " << idx << " is not implemented"
|
|
<< std::endl;
|
|
continue;
|
|
}
|
|
|
|
event_capture ec({
|
|
"drive_mounted",
|
|
"drive_unmounted",
|
|
"drive_unmount_pending",
|
|
"drive_mount_result",
|
|
});
|
|
|
|
std::thread th([&] {
|
|
const auto mounted = ec.wait_for_event("drive_mounted");
|
|
EXPECT_TRUE(mounted);
|
|
if (mounted) {
|
|
EXPECT_EQ(0,
|
|
system(("mount|grep \"" + mount_location + "\"").c_str()));
|
|
|
|
auto file_path = create_file_and_test(mount_location, "chmod_test");
|
|
test_chmod(utils::path::create_api_path("chmod_test"), file_path,
|
|
*provider_ptr);
|
|
unlink_file_and_test(file_path);
|
|
|
|
file_path = create_file_and_test(mount_location, "chown_test");
|
|
test_chown(utils::path::create_api_path("chown_test"), file_path,
|
|
*provider_ptr);
|
|
unlink_file_and_test(file_path);
|
|
|
|
file_path = utils::path::combine(mount_location, {"mkdir_test"});
|
|
test_mkdir(utils::path::create_api_path("mkdir_test"), file_path,
|
|
*provider_ptr);
|
|
rmdir_and_test(file_path);
|
|
|
|
file_path = create_file_and_test(mount_location, "write_read_test");
|
|
test_write_and_read(utils::path::create_api_path("write_read_test"),
|
|
file_path, *provider_ptr);
|
|
unlink_file_and_test(file_path);
|
|
|
|
file_path =
|
|
create_file_and_test(mount_location, "from_rename_file_test");
|
|
auto to_file_path = utils::path::absolute(
|
|
utils::path::combine(mount_location, {"to_rename_file_test"}));
|
|
test_rename_file(file_path, to_file_path, *provider_ptr);
|
|
EXPECT_TRUE(utils::file::retry_delete_file(file_path));
|
|
EXPECT_TRUE(utils::file::retry_delete_file(to_file_path));
|
|
|
|
file_path = utils::path::absolute(
|
|
utils::path::combine(mount_location, {"from_rename_dir_test"}));
|
|
to_file_path = utils::path::absolute(
|
|
utils::path::combine(mount_location, {"to_rename_dir_test"}));
|
|
test_rename_directory(file_path, to_file_path, *provider_ptr);
|
|
EXPECT_TRUE(utils::file::retry_delete_directory(file_path.c_str()));
|
|
EXPECT_TRUE(
|
|
utils::file::retry_delete_directory(to_file_path.c_str()));
|
|
|
|
file_path =
|
|
create_file_and_test(mount_location, "truncate_file_test");
|
|
test_truncate(file_path);
|
|
unlink_file_and_test(file_path);
|
|
|
|
file_path =
|
|
create_file_and_test(mount_location, "ftruncate_file_test");
|
|
test_ftruncate(file_path);
|
|
unlink_file_and_test(file_path);
|
|
|
|
#ifndef __APPLE__
|
|
file_path =
|
|
create_file_and_test(mount_location, "fallocate_file_test");
|
|
test_fallocate(utils::path::create_api_path("fallocate_file_test"),
|
|
file_path, *provider_ptr);
|
|
unlink_file_and_test(file_path);
|
|
#endif
|
|
|
|
file_path =
|
|
create_file_and_test(mount_location, "write_fails_ro_test");
|
|
test_write_operations_fail_if_read_only(
|
|
utils::path::create_api_path("write_fails_ro_test"), file_path,
|
|
*provider_ptr);
|
|
unlink_file_and_test(file_path);
|
|
|
|
file_path = create_file_and_test(mount_location, "getattr.txt");
|
|
test_file_getattr(utils::path::create_api_path("getattr.txt"),
|
|
file_path, *provider_ptr);
|
|
unlink_file_and_test(file_path);
|
|
|
|
file_path = utils::path::combine(mount_location, {"getattr_dir"});
|
|
test_directory_getattr(utils::path::create_api_path("getattr_dir"),
|
|
file_path, *provider_ptr);
|
|
rmdir_and_test(file_path);
|
|
|
|
#if !__APPLE__ && HAS_SETXATTR
|
|
file_path =
|
|
create_file_and_test(mount_location, "xattr_invalid_names_test");
|
|
test_xattr_invalid_parameters(file_path);
|
|
unlink_file_and_test(file_path);
|
|
|
|
file_path =
|
|
create_file_and_test(mount_location, "xattr_create_get_test");
|
|
test_xattr_create_and_get(file_path);
|
|
unlink_file_and_test(file_path);
|
|
|
|
file_path =
|
|
create_file_and_test(mount_location, "xattr_listxattr_test");
|
|
test_xattr_listxattr(file_path);
|
|
unlink_file_and_test(file_path);
|
|
|
|
file_path =
|
|
create_file_and_test(mount_location, "xattr_replace_test");
|
|
test_xattr_replace(file_path);
|
|
unlink_file_and_test(file_path);
|
|
|
|
file_path =
|
|
create_file_and_test(mount_location, "xattr_default_create_test");
|
|
test_xattr_default_create(file_path);
|
|
unlink_file_and_test(file_path);
|
|
|
|
file_path = create_file_and_test(mount_location,
|
|
"xattr_default_replace_test");
|
|
test_xattr_default_replace(file_path);
|
|
unlink_file_and_test(file_path);
|
|
|
|
file_path = create_file_and_test(mount_location,
|
|
"xattr_create_fails_exists_test");
|
|
test_xattr_create_fails_if_exists(file_path);
|
|
unlink_file_and_test(file_path);
|
|
|
|
file_path = create_file_and_test(
|
|
mount_location, "xattr_create_fails_not_exists_test");
|
|
test_xattr_create_fails_if_not_exists(file_path);
|
|
unlink_file_and_test(file_path);
|
|
|
|
file_path =
|
|
create_file_and_test(mount_location, "xattr_removexattr_test");
|
|
test_xattr_removexattr(file_path);
|
|
unlink_file_and_test(file_path);
|
|
#endif
|
|
}
|
|
|
|
execute_unmount(*provider_ptr, mount_location);
|
|
});
|
|
|
|
auto drive_args =
|
|
std::vector<std::string>({"unittests", "-f", mount_location});
|
|
execute_mount(*config_ptr, *lock_data_ptr, *provider_ptr, drive_args, th);
|
|
|
|
ec.wait_for_empty();
|
|
}
|
|
|
|
EXPECT_TRUE(utils::file::delete_directory_recursively(mount_location));
|
|
EXPECT_TRUE(utils::file::delete_directory_recursively(test_directory));
|
|
}
|
|
|
|
std::filesystem::current_path(current_directory);
|
|
}
|
|
} // namespace repertory
|
|
|
|
#endif
|