/* Copyright <2018-2025> 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. */ #ifndef FIFTHGRID_INCLUDE_UTILS_PATH_HPP_ #define FIFTHGRID_INCLUDE_UTILS_PATH_HPP_ #include "utils/config.hpp" #include "utils/string.hpp" namespace fifthgrid::utils::path { inline constexpr std::string_view backslash{"\\"}; inline constexpr std::wstring_view backslash_w{L"\\"}; inline constexpr std::string_view dot{"."}; inline constexpr std::wstring_view dot_w{L"."}; inline constexpr std::string_view dot_backslash{".\\"}; inline constexpr std::wstring_view dot_backslash_w{L".\\"}; inline constexpr std::string_view dot_slash{"./"}; inline constexpr std::wstring_view dot_slash_w{L"./"}; inline constexpr std::string_view long_notation{"\\\\?\\"}; inline constexpr std::wstring_view long_notation_w{L"\\\\?\\"}; inline constexpr std::string_view slash{"/"}; inline constexpr std::wstring_view slash_w{L"/"}; #if defined(_WIN32) inline constexpr std::string_view directory_seperator{backslash}; inline constexpr std::wstring_view directory_seperator_w{backslash_w}; inline constexpr std::string_view not_directory_seperator{slash}; inline constexpr std::wstring_view not_directory_seperator_w{slash_w}; inline constexpr std::string_view unc_notation{"\\\\"}; inline constexpr std::wstring_view unc_notation_w{L"\\\\"}; #else // !defined(_WIN32) inline constexpr std::string_view directory_seperator{slash}; inline constexpr std::wstring_view directory_seperator_w{slash_w}; inline constexpr std::string_view not_directory_seperator{backslash}; inline constexpr std::wstring_view not_directory_seperator_w{backslash_w}; #endif // defined(_WIN32) template [[nodiscard]] inline constexpr auto get_backslash() -> std::basic_string_view; template <> [[nodiscard]] inline constexpr auto get_backslash() -> std::basic_string_view { return backslash; } template <> [[nodiscard]] inline constexpr auto get_backslash() -> std::basic_string_view { return backslash_w; } template [[nodiscard]] inline constexpr auto get_directory_seperator() -> std::basic_string_view; template <> [[nodiscard]] inline constexpr auto get_directory_seperator() -> std::basic_string_view { return directory_seperator; } template <> [[nodiscard]] inline constexpr auto get_directory_seperator() -> std::basic_string_view { return directory_seperator_w; } template [[nodiscard]] inline constexpr auto get_not_directory_seperator() -> std::basic_string_view; template <> [[nodiscard]] inline constexpr auto get_not_directory_seperator() -> std::basic_string_view { return not_directory_seperator; } template <> [[nodiscard]] inline constexpr auto get_not_directory_seperator() -> std::basic_string_view { return not_directory_seperator_w; } template [[nodiscard]] inline constexpr auto get_dot() -> std::basic_string_view; template <> [[nodiscard]] inline constexpr auto get_dot() -> std::basic_string_view { return dot; } template <> [[nodiscard]] inline constexpr auto get_dot() -> std::basic_string_view { return dot_w; } template [[nodiscard]] inline constexpr auto get_dot_backslash() -> std::basic_string_view; template <> [[nodiscard]] inline constexpr auto get_dot_backslash() -> std::basic_string_view { return dot_backslash; } template <> [[nodiscard]] inline constexpr auto get_dot_backslash() -> std::basic_string_view { return dot_backslash_w; } template [[nodiscard]] inline constexpr auto get_dot_slash() -> std::basic_string_view; template <> [[nodiscard]] inline constexpr auto get_dot_slash() -> std::basic_string_view { return dot_slash; } template <> [[nodiscard]] inline constexpr auto get_dot_slash() -> std::basic_string_view { return dot_slash_w; } template [[nodiscard]] inline constexpr auto get_long_notation() -> std::basic_string_view; template <> [[nodiscard]] inline constexpr auto get_long_notation() -> std::basic_string_view { return long_notation; } template <> [[nodiscard]] inline constexpr auto get_long_notation() -> std::basic_string_view { return long_notation_w; } template [[nodiscard]] inline constexpr auto get_slash() -> std::basic_string_view; template <> [[nodiscard]] inline constexpr auto get_slash() -> std::basic_string_view { return slash; } template <> [[nodiscard]] inline constexpr auto get_slash() -> std::basic_string_view { return slash_w; } #if defined(_WIN32) template [[nodiscard]] inline constexpr auto get_unc_notation() -> std::basic_string_view; template <> [[nodiscard]] inline constexpr auto get_unc_notation() -> std::basic_string_view { return unc_notation; } template <> [[nodiscard]] inline constexpr auto get_unc_notation() -> std::basic_string_view { return unc_notation_w; } #endif // defined(_WIN32) template [[nodiscard]] inline auto get_current_path() -> string_t; [[nodiscard]] auto absolute(std::string_view path) -> std::string; [[nodiscard]] auto absolute(std::wstring_view path) -> std::wstring; [[nodiscard]] inline auto combine(std::string_view path, const std::vector &paths) -> std::string; [[nodiscard]] inline auto combine(std::wstring_view path, const std::vector &paths) -> std::wstring; [[nodiscard]] auto contains_trash_directory(std::string_view path) -> bool; [[nodiscard]] auto contains_trash_directory(std::wstring_view path) -> bool; [[nodiscard]] auto inline create_api_path(std::string_view path) -> std::string; [[nodiscard]] auto inline create_api_path(std::wstring_view path) -> std::wstring; [[nodiscard]] auto exists(std::string_view path) -> bool; [[nodiscard]] auto exists(std::wstring_view path) -> bool; [[nodiscard]] inline auto finalize(std::string_view path) -> std::string; [[nodiscard]] inline auto finalize(std::wstring_view path) -> std::wstring; [[nodiscard]] auto find_program_in_path(const std::string &name_without_extension) -> std::string; [[nodiscard]] auto find_program_in_path(std::wstring_view name_without_extension) -> std::wstring; template inline auto format_path(string_t &path, std::basic_string_view sep, std::basic_string_view not_sep) -> string_t &; [[nodiscard]] inline auto get_parent_api_path(std::string_view path) -> std::string; [[nodiscard]] inline auto get_parent_api_path(std::wstring_view path) -> std::wstring; [[nodiscard]] auto get_parent_path(std::string_view path) -> std::string; [[nodiscard]] auto get_parent_path(std::wstring_view path) -> std::wstring; [[nodiscard]] inline auto get_parts(std::string_view path) -> std::vector; [[nodiscard]] inline auto get_parts_w(std::wstring_view path) -> std::vector; [[nodiscard]] auto get_relative_path(std::string_view path, std::string_view root_path) -> std::string; [[nodiscard]] auto get_relative_path(std::wstring_view path, std::wstring_view root_path) -> std::wstring; [[nodiscard]] auto make_file_uri(std::string_view path) -> std::string; [[nodiscard]] auto make_file_uri(std::wstring_view path) -> std::wstring; [[nodiscard]] auto strip_to_file_name(std::string path) -> std::string; [[nodiscard]] auto strip_to_file_name(std::wstring path) -> std::wstring; [[nodiscard]] auto unmake_file_uri(std::string_view uri) -> std::string; [[nodiscard]] auto unmake_file_uri(std::wstring_view uri) -> std::wstring; template [[nodiscard]] inline auto combine_t( std::basic_string_view path, const std::vector> &paths) -> string_t { auto dir_sep_t = string_t{get_directory_seperator()}; return absolute( std::accumulate(paths.begin(), paths.end(), std::basic_string{path}, [&dir_sep_t](auto next_path, auto &&path_part) { if (next_path.empty()) { return string_t{path_part}; } return next_path + dir_sep_t + string_t{path_part}; })); } inline auto combine(std::string_view path, const std::vector &paths) -> std::string { return combine_t(path, paths); } inline auto combine(std::wstring_view path, const std::vector &paths) -> std::wstring { return combine_t(path, paths); } template [[nodiscard]] inline auto create_api_path_t( std::basic_string_view path) -> string_t { auto backslash_t = get_backslash(); auto dot_backslash_t = get_dot_backslash(); auto dot_slash_t = get_dot_slash(); auto dot_t = get_dot(); auto slash_t = get_slash(); #if defined(_WIN32) auto long_notation_t = get_long_notation(); if (utils::string::begins_with(path, long_notation_t)) { path = path.substr(long_notation_t.size()); } #endif // defined(_WIN32) if (path.empty() || path == backslash_t || path == dot_t || path == dot_slash_t || path == slash_t || path == dot_backslash_t) { return string_t{slash_t}; } string_t api_path{path}; #if defined(_WIN32) if ((api_path.size() >= 2U) && (api_path.at(1U) == ':')) { api_path = api_path.substr(2U); } #endif // defined(_WIN32) format_path(api_path, slash_t, backslash_t); while (utils::string::begins_with(api_path, dot_slash_t)) { api_path = api_path.substr(dot_slash_t.size()); } if (api_path.at(0U) != slash_t.at(0U)) { return string_t{slash_t} + api_path; } return api_path; } inline auto create_api_path(std::string_view path) -> std::string { return create_api_path_t(path); } inline auto create_api_path(std::wstring_view path) -> std::wstring { return create_api_path_t(path); } template [[nodiscard]] inline auto finalize_t( std::basic_string_view path) -> string_t { string_t dir_sep_t{get_directory_seperator()}; string_t fmt_path{path}; if (fmt_path.empty()) { return fmt_path; } format_path(fmt_path, dir_sep_t, get_not_directory_seperator()); #if defined(_WIN32) auto unc_notation_t = get_unc_notation(); if (utils::string::begins_with(fmt_path, unc_notation_t)) { return fmt_path; } auto dot_t = get_dot(); auto dot_sep_t = string_t{dot_t} + dir_sep_t; if (fmt_path == dot_t || fmt_path == dot_sep_t) { return get_current_path(); } if (fmt_path == dir_sep_t) { #if defined(PROJECT_ENABLE_WIN32_LONG_PATH_NAMES) return get_current_path().substr(0U, long_notation.size() + 2U); #else // !defined(PROJECT_ENABLE_WIN32_LONG_PATH_NAMES) return get_current_path().substr(0U, 2U); #endif // defined(PROJECT_ENABLE_WIN32_LONG_PATH_NAMES) } if (utils::string::begins_with(fmt_path, dir_sep_t)) { #if defined(PROJECT_ENABLE_WIN32_LONG_PATH_NAMES) return get_current_path().substr(0U, long_notation.size() + 2U) + fmt_path; #else // !defined(PROJECT_ENABLE_WIN32_LONG_PATH_NAMES) return get_current_path().substr(0U, 2U) + fmt_path; #endif // defined(PROJECT_ENABLE_WIN32_LONG_PATH_NAMES) } if (utils::string::begins_with(fmt_path, dot_sep_t)) { return get_current_path() + dir_sep_t + fmt_path.substr(2U); } #if defined(PROJECT_ENABLE_WIN32_LONG_PATH_NAMES) return string_t{get_long_notation()} + fmt_path; #endif // defined(PROJECT_ENABLE_WIN32_LONG_PATH_NAMES) #endif // defined(_WIN32) return fmt_path; } inline auto finalize(std::string_view path) -> std::string { return finalize_t(path); } inline auto finalize(std::wstring_view path) -> std::wstring { return finalize_t(path); } template inline auto format_path(string_t &path, std::basic_string_view sep, std::basic_string_view not_sep) -> string_t & { utils::string::replace(path, not_sep, sep); #if defined(_WIN32) auto is_unc{false}; auto long_notation_t = get_long_notation(); auto unc_notation_t = get_unc_notation(); if (utils::string::begins_with(path, long_notation_t)) { path = path.substr(long_notation_t.size()); } else if (utils::string::begins_with(path, unc_notation_t)) { path = path.substr(unc_notation_t.size()); utils::string::left_trim(path, sep.at(0U)); is_unc = true; } #endif // defined(_WIN32) string_t double_sep(2U, sep.at(0U)); while (utils::string::contains(path, double_sep)) { utils::string::replace(path, double_sep, sep); } if (path != sep) { utils::string::right_trim(path, sep.at(0U)); } #if defined(_WIN32) if (is_unc) { path = string_t{unc_notation_t} + path; } else if ((path.size() >= 2U) && (path.at(1U) == ':')) { path[0U] = utils::string::to_lower(string_t(1U, path.at(0U))).at(0U); } #endif // defined(_WIN32) return path; } template <> [[nodiscard]] inline auto get_current_path() -> std::string { #if defined(_WIN32) std::string path; path.resize(fifthgrid::max_path_length + 1U); ::GetCurrentDirectoryA(static_cast(path.size()), path.data()); path = path.c_str(); return finalize(path); #else // !defined(_WIN32) return finalize(std::filesystem::current_path().string()); #endif // defined(_WIN32) } template <> [[nodiscard]] inline auto get_current_path() -> std::wstring { return utils::string::from_utf8(get_current_path()); } template [[nodiscard]] inline auto get_parent_api_path_t( std::basic_string_view path) -> string_t { auto slash_t = get_slash(); string_t ret{path}; utils::string::right_trim(ret, slash_t.at(0U)); if (ret == slash_t || ret.empty()) { return string_t{path}; } auto sub_path = ret.substr(0, ret.rfind(slash_t) + 1); if (sub_path == slash_t) { return string_t{sub_path}; } return sub_path; } inline auto get_parent_api_path(std::string_view path) -> std::string { return create_api_path(get_parent_api_path_t(path)); } inline auto get_parent_api_path(std::wstring_view path) -> std::wstring { return create_api_path(get_parent_api_path_t(path)); } template [[nodiscard]] inline auto get_parts_t(std::basic_string_view path) -> std::vector { return utils::string::split( path, get_directory_seperator().at(0U), false); } inline auto get_parts(std::string_view path) -> std::vector { return get_parts_t(path); } inline auto get_parts_w(std::wstring_view path) -> std::vector { return get_parts_t(path); } } // namespace fifthgrid::utils::path #endif // FIFTHGRID_INCLUDE_UTILS_PATH_HPP_