Compare commits
	
		
			23 Commits
		
	
	
		
			v2.0.5-rc
			...
			98edf33be4
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 98edf33be4 | |||
| a080c9ff86 | |||
| e6cdcd74a1 | |||
| 573ae549be | |||
| fca149f998 | |||
| 76e375c488 | |||
| 3f9322f659 | |||
| c286d496c3 | |||
| 56ba0fcb83 | |||
| dcafb104ea | |||
| ab757dfd36 | |||
| eec2d2e9a9 | |||
| 8c1c91e02b | |||
| bb85015733 | |||
| 883de836c6 | |||
| bf2bdd1b5d | |||
| f1e82d8f9f | |||
| f5c4aebdac | |||
| f2f9e8fd15 | |||
| 2a673915af | |||
| df3db38ae7 | |||
| b0b69c6dd4 | |||
| 11b118a30f | 
							
								
								
									
										16
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										16
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -1,7 +1,23 @@ | ||||
| # Changelog | ||||
|  | ||||
| ## v2.0.6-release | ||||
|  | ||||
| ### Issues | ||||
|  | ||||
| * ~~\#12 [Unit Test] Complete all providers unit tests~~ | ||||
| * ~~\#21 [Unit Test] Complete WinFSP unit tests~~ | ||||
| * ~~\#22 [Unit Test] Complete FUSE unit tests~~ | ||||
| * ~~\#33 Complete initial v2.0 documentation~~ | ||||
| * \#42 [bug] Remote mount directory listing on Windows connected to Linux is failing | ||||
| * \#43 [bug] Directories are not importing properly for Sia | ||||
|  | ||||
| ### Changes from v2.0.5-rc | ||||
|  | ||||
| * Drive letters in UI should always be lowercase | ||||
|  | ||||
| ## v2.0.5-rc | ||||
|  | ||||
| <!-- markdownlint-disable-next-line --> | ||||
| ### Issues | ||||
|  | ||||
| * \#39 Create management portal in Flutter | ||||
|   | ||||
| @@ -10,7 +10,7 @@ PROJECT_DESC="Mount utility for Sia and S3" | ||||
|  | ||||
| PROJECT_MAJOR_VERSION=2 | ||||
| PROJECT_MINOR_VERSION=0 | ||||
| PROJECT_REVISION_VERSION=5 | ||||
| PROJECT_REVISION_VERSION=6 | ||||
| PROJECT_RELEASE_NUM=0 | ||||
| PROJECT_RELEASE_ITER=rc | ||||
|  | ||||
|   | ||||
| @@ -225,8 +225,7 @@ public: | ||||
|     } | ||||
|  | ||||
|     if (curl_code != CURLE_OK) { | ||||
|       event_system::instance().raise<curl_error>(curl_code, function_name, | ||||
|                                                   url); | ||||
|       event_system::instance().raise<curl_error>(curl_code, function_name, url); | ||||
|       return false; | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -59,7 +59,8 @@ private: | ||||
|  | ||||
|   static void populate_stat(const struct stat64 &unix_st, remote::stat &r_stat); | ||||
|  | ||||
|   [[nodiscard]] auto update_to_windows_format(json &item) -> json &; | ||||
|   [[nodiscard]] auto update_to_windows_format(const std::string &root_api_path, | ||||
|                                               json &item) -> json &; | ||||
|  | ||||
| public: | ||||
|   // FUSE Layer | ||||
|   | ||||
| @@ -49,6 +49,9 @@ private: | ||||
|   sia_config sia_config_; | ||||
|  | ||||
| private: | ||||
|   [[nodiscard]] auto create_directory_key(const std::string &api_path) const | ||||
|       -> repertory::api_error; | ||||
|  | ||||
|   [[nodiscard]] auto get_object_info(const std::string &api_path, | ||||
|                                      json &object_info) const -> api_error; | ||||
|  | ||||
|   | ||||
| @@ -374,13 +374,13 @@ auto fuse_base::init_impl(struct fuse_conn_info *conn) -> void * { | ||||
|   if (not utils::file::change_to_process_directory()) { | ||||
|     utils::error::raise_error(function_name, | ||||
|                               "failed to change to process directory"); | ||||
|     event_system::instance().raise<unmount_requested>(); | ||||
|     event_system::instance().raise<unmount_requested>(function_name); | ||||
|     return this; | ||||
|   } | ||||
|  | ||||
|   if (not console_enabled_ && not repertory::project_initialize()) { | ||||
|     utils::error::raise_error(function_name, "failed to initialize repertory"); | ||||
|     event_system::instance().raise<unmount_requested>(); | ||||
|     event_system::instance().raise<unmount_requested>(function_name); | ||||
|   } | ||||
|  | ||||
|   return this; | ||||
|   | ||||
| @@ -259,7 +259,7 @@ auto remote_fuse_drive::init_impl(struct fuse_conn_info *conn) -> void * { | ||||
|   if (remote_instance_->fuse_init() != 0) { | ||||
|     utils::error::raise_error(function_name, | ||||
|                               "failed to connect to remote server"); | ||||
|     event_system::instance().raise<unmount_requested>(); | ||||
|     event_system::instance().raise<unmount_requested>(function_name); | ||||
|   } else { | ||||
|     server_ = std::make_shared<server>(config_); | ||||
|     server_->start(); | ||||
|   | ||||
| @@ -1390,11 +1390,11 @@ auto remote_server::winfsp_read_directory(PVOID file_desc, PWSTR /*pattern*/, | ||||
|  | ||||
|   item_list.clear(); | ||||
|  | ||||
|   const auto handle = reinterpret_cast<remote::file_handle>(file_desc); | ||||
|   auto handle = reinterpret_cast<remote::file_handle>(file_desc); | ||||
|   auto ret = static_cast<packet::error_type>( | ||||
|       has_open_info(static_cast<native_handle>(handle), STATUS_INVALID_HANDLE)); | ||||
|   if (ret == STATUS_SUCCESS) { | ||||
|     const auto api_path = construct_api_path( | ||||
|     auto api_path = construct_api_path( | ||||
|         get_open_file_path(static_cast<native_handle>(handle))); | ||||
|     directory_iterator iterator(drive_.get_directory_items(api_path)); | ||||
|     auto offset = marker == nullptr | ||||
| @@ -1405,7 +1405,7 @@ auto remote_server::winfsp_read_directory(PVOID file_desc, PWSTR /*pattern*/, | ||||
|     json item; | ||||
|     while (iterator.get_json(offset++, item) == 0) { | ||||
|       try { | ||||
|         item_list.emplace_back(update_to_windows_format(item)); | ||||
|         item_list.emplace_back(update_to_windows_format(api_path, item)); | ||||
|       } catch (const std::exception &e) { | ||||
|         utils::error::raise_error(function_name, e, "exception occurred"); | ||||
|       } | ||||
| @@ -1679,30 +1679,53 @@ auto remote_server::json_release_directory_snapshot( | ||||
|   return 0; | ||||
| } | ||||
|  | ||||
| auto remote_server::update_to_windows_format(json &item) -> json & { | ||||
|   const auto api_path = item["path"].get<std::string>(); | ||||
|   item["meta"][META_ACCESSED] = std::to_string( | ||||
|       utils::string::to_uint64(empty_as_zero(item["meta"][META_ACCESSED]))); | ||||
|   item["meta"][META_CREATION] = std::to_string( | ||||
|       utils::string::to_uint64(empty_as_zero(item["meta"][META_CREATION]))); | ||||
|   item["meta"][META_MODIFIED] = std::to_string( | ||||
|       utils::string::to_uint64(empty_as_zero(item["meta"][META_MODIFIED]))); | ||||
| auto remote_server::update_to_windows_format(const std::string &root_api_path, | ||||
|                                              json &item) -> json & { | ||||
|   REPERTORY_USES_FUNCTION_NAME(); | ||||
|  | ||||
|   if (item["meta"][META_WRITTEN].empty() || | ||||
|       (item["meta"][META_WRITTEN].get<std::string>() == "0") || | ||||
|       (item["meta"][META_WRITTEN].get<std::string>() == | ||||
|        std::to_string(utils::time::WIN32_TIME_CONVERSION))) { | ||||
|     drive_.set_item_meta(api_path, META_WRITTEN, | ||||
|                          item["meta"][META_MODIFIED].get<std::string>()); | ||||
|     item["meta"][META_WRITTEN] = item["meta"][META_MODIFIED]; | ||||
|   auto api_path = item[JSON_API_PATH].get<std::string>(); | ||||
|   if (api_path == "." || api_path == "..") { | ||||
|     auto orig_api_path{api_path}; | ||||
|  | ||||
|     api_path = api_path == "." | ||||
|                    ? root_api_path | ||||
|                    : utils::path::get_parent_api_path(root_api_path); | ||||
|  | ||||
|     api_meta_map meta; | ||||
|     auto res = drive_.get_item_meta(api_path, meta); | ||||
|     if (res != api_error::success) { | ||||
|       utils::error::raise_api_path_error( | ||||
|           function_name, api_path, res, | ||||
|           fmt::format("failed to get '{}' meta", orig_api_path)); | ||||
|       return item; | ||||
|     } | ||||
|  | ||||
|     item[JSON_META] = meta; | ||||
|   } | ||||
|  | ||||
|   if (item["meta"][META_ATTRIBUTES].empty()) { | ||||
|     item["meta"][META_ATTRIBUTES] = | ||||
|         item["directory"].get<bool>() ? std::to_string(FILE_ATTRIBUTE_DIRECTORY) | ||||
|                                       : std::to_string(FILE_ATTRIBUTE_ARCHIVE); | ||||
|   item[JSON_META][META_ACCESSED] = std::to_string( | ||||
|       utils::string::to_uint64(empty_as_zero(item[JSON_META][META_ACCESSED]))); | ||||
|   item[JSON_META][META_CREATION] = std::to_string( | ||||
|       utils::string::to_uint64(empty_as_zero(item[JSON_META][META_CREATION]))); | ||||
|   item[JSON_META][META_MODIFIED] = std::to_string( | ||||
|       utils::string::to_uint64(empty_as_zero(item[JSON_META][META_MODIFIED]))); | ||||
|  | ||||
|   if (item[JSON_META][META_WRITTEN].empty() || | ||||
|       (item[JSON_META][META_WRITTEN].get<std::string>() == "0") || | ||||
|       (item[JSON_META][META_WRITTEN].get<std::string>() == | ||||
|        std::to_string(utils::time::WIN32_TIME_CONVERSION))) { | ||||
|     drive_.set_item_meta(api_path, META_WRITTEN, | ||||
|                          item[JSON_META][META_MODIFIED].get<std::string>()); | ||||
|     item[JSON_META][META_WRITTEN] = item[JSON_META][META_MODIFIED]; | ||||
|   } | ||||
|  | ||||
|   if (item[JSON_META][META_ATTRIBUTES].empty()) { | ||||
|     item[JSON_META][META_ATTRIBUTES] = | ||||
|         item[JSON_DIRECTORY].get<bool>() | ||||
|             ? std::to_string(FILE_ATTRIBUTE_DIRECTORY) | ||||
|             : std::to_string(FILE_ATTRIBUTE_ARCHIVE); | ||||
|     drive_.set_item_meta(api_path, META_ATTRIBUTES, | ||||
|                          item["meta"][META_ATTRIBUTES].get<std::string>()); | ||||
|                          item[JSON_META][META_ATTRIBUTES].get<std::string>()); | ||||
|   } | ||||
|  | ||||
|   return item; | ||||
|   | ||||
| @@ -361,11 +361,10 @@ auto remote_winfsp_drive::ReadDirectory(PVOID /*file_node*/, PVOID file_desc, | ||||
|               &ret)) { | ||||
|         auto item_found = false; | ||||
|         for (const auto &item : item_list) { | ||||
|           auto item_path = item["path"].get<std::string>(); | ||||
|           auto item_path = item[JSON_API_PATH].get<std::string>(); | ||||
|           auto display_name = utils::string::from_utf8( | ||||
|               utils::path::strip_to_file_name(item_path)); | ||||
|           if (not marker || (marker && item_found)) { | ||||
|             // if (not utils::path::is_ads_file_path(item_path)) { | ||||
|             union { | ||||
|               UINT8 B[FIELD_OFFSET(FSP_FSCTL_DIR_INFO, FileNameBuf) + | ||||
|                       ((repertory::max_path_length + 1U) * sizeof(WCHAR))]; | ||||
| @@ -380,10 +379,8 @@ auto remote_winfsp_drive::ReadDirectory(PVOID /*file_node*/, PVOID file_desc, | ||||
|                           display_name.size()) * | ||||
|                  sizeof(WCHAR))); | ||||
|  | ||||
|             if (not item["meta"].empty() || | ||||
|                 ((item_path != ".") && (item_path != ".."))) { | ||||
|               populate_file_info(item, directory_info->FileInfo); | ||||
|             } | ||||
|             populate_file_info(item, directory_info->FileInfo); | ||||
|  | ||||
|             if (ret == STATUS_SUCCESS) { | ||||
|               ::wcscpy_s(&directory_info->FileNameBuf[0], | ||||
|                          repertory::max_path_length, &display_name[0]); | ||||
| @@ -394,7 +391,6 @@ auto remote_winfsp_drive::ReadDirectory(PVOID /*file_node*/, PVOID file_desc, | ||||
|                 break; | ||||
|               } | ||||
|             } | ||||
|             // } | ||||
|           } else { | ||||
|             item_found = display_name == std::wstring(marker); | ||||
|           } | ||||
|   | ||||
| @@ -149,6 +149,31 @@ auto sia_provider::create_directory_impl(const std::string &api_path, | ||||
|   return api_error::success; | ||||
| } | ||||
|  | ||||
| auto sia_provider::create_directory_key(const std::string &api_path) const | ||||
|     -> api_error { | ||||
|   auto parent_api_path = utils::path::get_parent_api_path(api_path); | ||||
|  | ||||
|   json object_list{}; | ||||
|   if (not get_object_list(parent_api_path, object_list)) { | ||||
|     return api_error::comm_error; | ||||
|   } | ||||
|  | ||||
|   if (not object_list.contains("objects")) { | ||||
|     return api_error::item_not_found; | ||||
|   } | ||||
|  | ||||
|   const auto &list = object_list.at("objects"); | ||||
|   if (std::ranges::find_if(list, [&api_path](auto &&entry) -> bool { | ||||
|         return entry.at("key").template get<std::string>() == api_path + '/'; | ||||
|       }) == list.end()) { | ||||
|     return api_error::item_not_found; | ||||
|   } | ||||
|  | ||||
|   api_meta_map meta; | ||||
|   return const_cast<sia_provider *>(this)->create_directory_impl(api_path, | ||||
|                                                                  meta); | ||||
| } | ||||
|  | ||||
| auto sia_provider::get_directory_item_count(const std::string &api_path) const | ||||
|     -> std::uint64_t { | ||||
|   REPERTORY_USES_FUNCTION_NAME(); | ||||
| @@ -320,6 +345,17 @@ auto sia_provider::get_file_list(api_file_list &list, | ||||
|                   create_api_file(entry_api_path, "", 0U, | ||||
|                                   get_last_modified(entry)), | ||||
|               }; | ||||
|  | ||||
|               bool exists{}; | ||||
|               auto res{is_directory(entry_api_path, exists)}; | ||||
|               if (res != api_error::success) { | ||||
|                 return res; | ||||
|               } | ||||
|  | ||||
|               if (not exists) { | ||||
|                 return api_error::directory_not_found; | ||||
|               } | ||||
|  | ||||
|               get_api_item_added()(true, dir); | ||||
|             } | ||||
|  | ||||
| @@ -512,6 +548,13 @@ auto sia_provider::is_directory(const std::string &api_path, bool &exists) const | ||||
|  | ||||
|     json file_data{}; | ||||
|     auto res{get_object_info(api_path + '/', file_data)}; | ||||
|     if (res == api_error::item_not_found) { | ||||
|       if (create_directory_key(api_path) == api_error::success) { | ||||
|         exists = true; | ||||
|         return api_error::success; | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     if (res == api_error::item_not_found) { | ||||
|       return api_error::success; | ||||
|     } | ||||
| @@ -636,7 +679,6 @@ auto sia_provider::read_file_bytes(const std::string &api_path, | ||||
|          ++idx) { | ||||
|       long response_code{}; | ||||
|       const auto notify_retry = [&]() { | ||||
|         fmt::println("{}", std::string(buffer.begin(), buffer.end())); | ||||
|         if (response_code == 0) { | ||||
|           utils::error::raise_api_path_error( | ||||
|               function_name, api_path, api_error::comm_error, | ||||
|   | ||||
| @@ -68,7 +68,9 @@ void server::handle_set_config_value_by_name(const httplib::Request &req, | ||||
|  | ||||
| void server::handle_unmount(const httplib::Request & /*req*/, | ||||
|                             httplib::Response &res) { | ||||
|   event_system::instance().raise<unmount_requested>(); | ||||
|   REPERTORY_USES_FUNCTION_NAME(); | ||||
|  | ||||
|   event_system::instance().raise<unmount_requested>(function_name); | ||||
|   res.status = http_error_codes::ok; | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -23,6 +23,8 @@ | ||||
| #include "backward.hpp" | ||||
| #endif // defined(PROJECT_ENABLE_BACKWARD_CPP) | ||||
|  | ||||
| #include <algorithm> | ||||
|  | ||||
| #include "cli/actions.hpp" | ||||
| #include "initialize.hpp" | ||||
| #include "types/repertory.hpp" | ||||
| @@ -44,7 +46,7 @@ auto main(int argc, char **argv) -> int { | ||||
|   std::vector<const char *> args; | ||||
|   { | ||||
|     auto args_span = std::span(argv, static_cast<std::size_t>(argc)); | ||||
|     std::copy(args_span.begin(), args_span.end(), std::back_inserter(args)); | ||||
|     std::ranges::copy(args_span, std::back_inserter(args)); | ||||
|   } | ||||
|  | ||||
|   if (argc == 1) { | ||||
|   | ||||
| @@ -326,9 +326,9 @@ void handlers::handle_put_mount_location(const httplib::Request &req, | ||||
| void handlers::handle_get_available_locations(httplib::Response &res) { | ||||
| #if defined(_WIN32) | ||||
|   constexpr const std::array<std::string_view, 26U> letters{ | ||||
|       "A:", "B:", "C:", "D:", "E:", "F:", "G:", "H:", "I:", | ||||
|       "J:", "K:", "L:", "M:", "N:", "O:", "P:", "Q:", "R:", | ||||
|       "S:", "T:", "U:", "V:", "W:", "X:", "Y:", "Z:", | ||||
|       "a:", "b:", "c:", "d:", "e:", "f:", "g:", "h:", "i:", | ||||
|       "j:", "k:", "l:", "m:", "n:", "o:", "p:", "q:", "r:", | ||||
|       "s:", "t:", "u:", "v:", "w:", "x:", "y:", "z:", | ||||
|   }; | ||||
|  | ||||
|   auto available = std::accumulate( | ||||
|   | ||||
| @@ -54,21 +54,31 @@ namespace repertory { | ||||
| struct local_s3 final { | ||||
|   static constexpr const provider_type type{provider_type::s3}; | ||||
|   static constexpr const provider_type type2{provider_type::s3}; | ||||
|   static constexpr const std::uint16_t port{0U}; | ||||
| }; | ||||
|  | ||||
| struct local_sia final { | ||||
|   static constexpr const provider_type type{provider_type::sia}; | ||||
|   static constexpr const provider_type type2{provider_type::sia}; | ||||
|   static constexpr const std::uint16_t port{0U}; | ||||
| }; | ||||
|  | ||||
| struct remote_s3 final { | ||||
|   static constexpr const provider_type type{provider_type::remote}; | ||||
|   static constexpr const provider_type type2{provider_type::s3}; | ||||
|   static constexpr const std::uint16_t port{0U}; | ||||
| }; | ||||
|  | ||||
| struct remote_sia final { | ||||
|   static constexpr const provider_type type{provider_type::remote}; | ||||
|   static constexpr const provider_type type2{provider_type::sia}; | ||||
|   static constexpr const std::uint16_t port{0U}; | ||||
| }; | ||||
|  | ||||
| struct remote_linux_to_winfsp final { | ||||
|   static constexpr const provider_type type{provider_type::remote}; | ||||
|   static constexpr const provider_type type2{provider_type::unknown}; | ||||
|   static constexpr const std::uint16_t port{40001U}; | ||||
| }; | ||||
|  | ||||
| template <typename provider_t> class fuse_test : public ::testing::Test { | ||||
| @@ -113,7 +123,7 @@ protected: | ||||
|  | ||||
|           auto r_cfg = config->get_remote_mount(); | ||||
|           r_cfg.enable = true; | ||||
|           r_cfg.api_port = 30000U; | ||||
|           r_cfg.api_port = 40000U; | ||||
|           config->set_remote_mount(r_cfg); | ||||
|         } | ||||
|  | ||||
| @@ -160,7 +170,7 @@ protected: | ||||
|  | ||||
|           auto r_cfg = config->get_remote_mount(); | ||||
|           r_cfg.enable = true; | ||||
|           r_cfg.api_port = 30000U; | ||||
|           r_cfg.api_port = 40000U; | ||||
|           config->set_remote_mount(r_cfg); | ||||
|         } | ||||
|  | ||||
| @@ -178,7 +188,7 @@ protected: | ||||
|       execute_mount(drive_args, mount_location); | ||||
|     }; | ||||
|  | ||||
|     const auto mount_remote = [&]() { | ||||
|     const auto mount_remote = [&](std::uint16_t port = 40000U) { | ||||
|       { | ||||
|         mount_location2 = mount_location; | ||||
|  | ||||
| @@ -187,7 +197,8 @@ protected: | ||||
|             { | ||||
|                 "fuse_test", | ||||
|                 app_config::get_provider_name(provider_t::type) + '_' + | ||||
|                     app_config::get_provider_name(provider_t::type2), | ||||
|                     app_config::get_provider_name(provider_t::type2) + '_' + | ||||
|                     std::to_string(port), | ||||
|             }); | ||||
|  | ||||
|         mount_location = utils::path::combine(test_directory, {"mount"}); | ||||
| @@ -206,7 +217,7 @@ protected: | ||||
|             "-dd", | ||||
|             config2->get_data_directory(), | ||||
|             "-rm", | ||||
|             "localhost:30000", | ||||
|             fmt::format("localhost:{}", port), | ||||
|             mount_location, | ||||
|         }); | ||||
|       } | ||||
| @@ -233,6 +244,10 @@ protected: | ||||
|         mount_sia(); | ||||
|       } break; | ||||
|  | ||||
|       case provider_type::unknown: | ||||
|         mount_remote(provider_t::port); | ||||
|         return; | ||||
|  | ||||
|       default: | ||||
|         throw std::runtime_error("remote provider type is not implemented"); | ||||
|         return; | ||||
| @@ -410,9 +425,15 @@ std::string fuse_test<provider_t>::mount_location; | ||||
| template <typename provider_t> | ||||
| std::string fuse_test<provider_t>::mount_location2; | ||||
|  | ||||
| // using fuse_provider_types = ::testing::Types<local_s3, remote_s3>; | ||||
| #if defined(__linux__) | ||||
| using fuse_provider_types = | ||||
|     ::testing::Types<local_s3, remote_s3, local_sia, remote_sia>; | ||||
| // using fuse_provider_types = | ||||
| //     ::testing::Types<local_s3, remote_s3, local_sia, remote_sia, | ||||
| //     remote_linux_to_winfsp>; | ||||
| #else  // !defined(__linux__) | ||||
| build fails here | ||||
| #endif // defined(_WIN32) | ||||
| } // namespace repertory | ||||
|  | ||||
| #endif // !defined(_WIN32) | ||||
|   | ||||
| @@ -45,21 +45,31 @@ namespace repertory { | ||||
| struct local_s3 final { | ||||
|   static constexpr const provider_type type{provider_type::s3}; | ||||
|   static constexpr const provider_type type2{provider_type::s3}; | ||||
|   static constexpr const std::uint16_t port{0U}; | ||||
| }; | ||||
|  | ||||
| struct local_sia final { | ||||
|   static constexpr const provider_type type{provider_type::sia}; | ||||
|   static constexpr const provider_type type2{provider_type::sia}; | ||||
|   static constexpr const std::uint16_t port{0U}; | ||||
| }; | ||||
|  | ||||
| struct remote_s3 final { | ||||
|   static constexpr const provider_type type{provider_type::remote}; | ||||
|   static constexpr const provider_type type2{provider_type::s3}; | ||||
|   static constexpr const std::uint16_t port{0U}; | ||||
| }; | ||||
|  | ||||
| struct remote_sia final { | ||||
|   static constexpr const provider_type type{provider_type::remote}; | ||||
|   static constexpr const provider_type type2{provider_type::sia}; | ||||
|   static constexpr const std::uint16_t port{0U}; | ||||
| }; | ||||
|  | ||||
| struct remote_winfsp_to_linux final { | ||||
|   static constexpr const provider_type type{provider_type::remote}; | ||||
|   static constexpr const provider_type type2{provider_type::unknown}; | ||||
|   static constexpr const std::uint16_t port{40001U}; | ||||
| }; | ||||
|  | ||||
| template <typename provider_t> class winfsp_test : public ::testing::Test { | ||||
| @@ -102,7 +112,7 @@ protected: | ||||
|  | ||||
|           auto r_cfg = config->get_remote_mount(); | ||||
|           r_cfg.enable = true; | ||||
|           r_cfg.api_port = 30000U; | ||||
|           r_cfg.api_port = 40000U; | ||||
|           config->set_remote_mount(r_cfg); | ||||
|         } | ||||
|  | ||||
| @@ -144,7 +154,7 @@ protected: | ||||
|  | ||||
|           auto r_cfg = config->get_remote_mount(); | ||||
|           r_cfg.enable = true; | ||||
|           r_cfg.api_port = 30000U; | ||||
|           r_cfg.api_port = 40000U; | ||||
|           config->set_remote_mount(r_cfg); | ||||
|         } | ||||
|  | ||||
| @@ -160,13 +170,15 @@ protected: | ||||
|       execute_mount(drive_args, mount_location); | ||||
|     }; | ||||
|  | ||||
|     const auto mount_remote = [&]() { | ||||
|     const auto mount_remote = [&](std::uint16_t port = 40000U) { | ||||
|       { | ||||
|         auto test_directory = utils::path::combine( | ||||
|             test::get_test_output_dir(), | ||||
|             { | ||||
|                 "winfsp_test", | ||||
|                 app_config::get_provider_name(provider_type::remote), | ||||
|                 app_config::get_provider_name(provider_t::type) + '_' + | ||||
|                     app_config::get_provider_name(provider_t::type2) + '_' + | ||||
|                     std::to_string(port), | ||||
|             }); | ||||
|  | ||||
|         mount_location2 = mount_location; | ||||
| @@ -184,7 +196,7 @@ protected: | ||||
|             "-dd", | ||||
|             config->get_data_directory(), | ||||
|             "-rm", | ||||
|             "localhost:30000", | ||||
|             fmt::format("localhost:{}", port), | ||||
|             mount_location, | ||||
|         }); | ||||
|       } | ||||
| @@ -211,6 +223,10 @@ protected: | ||||
|         mount_sia(); | ||||
|       } break; | ||||
|  | ||||
|       case provider_type::unknown: | ||||
|         mount_remote(provider_t::port); | ||||
|         return; | ||||
|  | ||||
|       default: | ||||
|         throw std::runtime_error("remote provider type is not implemented"); | ||||
|         return; | ||||
| @@ -228,7 +244,9 @@ protected: | ||||
|   static void TearDownTestCase() { | ||||
|     if (provider_t::type == provider_type::remote) { | ||||
|       execute_unmount(drive_args2, mount_location); | ||||
|       execute_unmount(drive_args, mount_location2); | ||||
|       if (provider_t::type2 != provider_type::unknown) { | ||||
|         execute_unmount(drive_args, mount_location2); | ||||
|       } | ||||
|     } else { | ||||
|       execute_unmount(drive_args, mount_location); | ||||
|     } | ||||
| @@ -282,7 +300,8 @@ std::string winfsp_test<provider_t>::mount_location2; | ||||
|  | ||||
| // using winfsp_provider_types = ::testing::Types<local_s3, remote_s3, | ||||
| // local_sia, remote_sia>; | ||||
| using winfsp_provider_types = ::testing::Types<local_s3, remote_s3>; | ||||
| // using winfsp_provider_types = ::testing::Types<local_s3, remote_s3>; | ||||
| using winfsp_provider_types = ::testing::Types<remote_winfsp_to_linux>; | ||||
| } // namespace repertory | ||||
|  | ||||
| #endif // defined(_WIN32) | ||||
|   | ||||
		Reference in New Issue
	
	Block a user