#pragma once #include #if USE_STD_FILESYSTEM #include #else #define _SILENCE_EXPERIMENTAL_FILESYSTEM_DEPRECATION_WARNING #include #endif namespace fs { #if USE_STD_FILESYSTEM namespace stdfs = std::filesystem; #else namespace stdfs = std::experimental::filesystem; #endif using stdfs::copy_options; using stdfs::path; using stdfs::perms; using stdfs::u8path; #if defined(_WIN32) enum class file_type { none = 0, not_found = -1, regular = 1, directory = 2, symlink = 3, block = 4, character = 5, fifo = 6, socket = 7, unknown = 8, // also stands for a junction directory_symlink = 42 }; struct file_status { explicit file_status(file_type type = file_type::none, perms permissions = perms::unknown) noexcept : m_type(type), m_permissions(permissions) { } file_type type() const noexcept { return m_type; } void type(file_type type) noexcept { m_type = type; } perms permissions() const noexcept { return m_permissions; } void permissions(perms perm) noexcept { m_permissions = perm; } private: file_type m_type; perms m_permissions; }; #else using stdfs::file_type; // to set up ADL correctly on `file_status` objects, we are defining // this in our own namespace struct file_status : private stdfs::file_status { using stdfs::file_status::file_status; using stdfs::file_status::permissions; using stdfs::file_status::type; }; #endif inline bool is_symlink(file_status s) noexcept { #if defined(_WIN32) if (s.type() == file_type::directory_symlink) return true; #endif return s.type() == file_type::symlink; } inline bool is_regular_file(file_status s) { return s.type() == file_type::regular; } inline bool is_directory(file_status s) { return s.type() == file_type::directory; } inline bool exists(file_status s) { return s.type() != file_type::not_found && s.type() != file_type::none; } } /* if someone attempts to use unqualified `symlink_status` or `is_symlink`, they might get the ADL version, which is broken. Therefore, put `(symlink_)?status` as deleted in the global namespace, so that they get an error. We also want to poison the ADL on the other functions, because we don't want people calling these functions on paths */ void status(const fs::path& p) = delete; void status(const fs::path& p, std::error_code& ec) = delete; void symlink_status(const fs::path& p) = delete; void symlink_status(const fs::path& p, std::error_code& ec) = delete; void is_symlink(const fs::path& p) = delete; void is_symlink(const fs::path& p, std::error_code& ec) = delete; void is_regular_file(const fs::path& p) = delete; void is_regular_file(const fs::path& p, std::error_code& ec) = delete; void is_directory(const fs::path& p) = delete; void is_directory(const fs::path& p, std::error_code& ec) = delete; namespace vcpkg::Files { struct Filesystem { std::string read_contents(const fs::path& file_path, LineInfo linfo) const; virtual Expected read_contents(const fs::path& file_path) const = 0; virtual Expected> read_lines(const fs::path& file_path) const = 0; virtual fs::path find_file_recursively_up(const fs::path& starting_dir, const std::string& filename) const = 0; virtual std::vector get_files_recursive(const fs::path& dir) const = 0; virtual std::vector get_files_non_recursive(const fs::path& dir) const = 0; void write_lines(const fs::path& file_path, const std::vector& lines, LineInfo linfo); virtual void write_lines(const fs::path& file_path, const std::vector& lines, std::error_code& ec) = 0; void write_contents(const fs::path& path, const std::string& data, LineInfo linfo); virtual void write_contents(const fs::path& file_path, const std::string& data, std::error_code& ec) = 0; void rename(const fs::path& oldpath, const fs::path& newpath, LineInfo linfo); virtual void rename(const fs::path& oldpath, const fs::path& newpath, std::error_code& ec) = 0; virtual void rename_or_copy(const fs::path& oldpath, const fs::path& newpath, StringLiteral temp_suffix, std::error_code& ec) = 0; bool remove(const fs::path& path, LineInfo linfo); virtual bool remove(const fs::path& path, std::error_code& ec) = 0; virtual void remove_all(const fs::path& path, std::error_code& ec, fs::path& failure_point) = 0; void remove_all(const fs::path& path, LineInfo li); bool exists(const fs::path& path, std::error_code& ec) const; bool exists(LineInfo li, const fs::path& path) const; // this should probably not exist, but would require a pass through of // existing code to fix bool exists(const fs::path& path) const; virtual bool is_directory(const fs::path& path) const = 0; virtual bool is_regular_file(const fs::path& path) const = 0; virtual bool is_empty(const fs::path& path) const = 0; virtual bool create_directory(const fs::path& path, std::error_code& ec) = 0; virtual bool create_directories(const fs::path& path, std::error_code& ec) = 0; virtual void copy(const fs::path& oldpath, const fs::path& newpath, fs::copy_options opts) = 0; virtual bool copy_file(const fs::path& oldpath, const fs::path& newpath, fs::copy_options opts, std::error_code& ec) = 0; virtual void copy_symlink(const fs::path& oldpath, const fs::path& newpath, std::error_code& ec) = 0; virtual fs::file_status status(const fs::path& path, std::error_code& ec) const = 0; virtual fs::file_status symlink_status(const fs::path& path, std::error_code& ec) const = 0; fs::file_status status(LineInfo li, const fs::path& p) const noexcept; fs::file_status symlink_status(LineInfo li, const fs::path& p) const noexcept; virtual fs::path canonical(const fs::path& path, std::error_code& ec) const = 0; fs::path canonical(LineInfo li, const fs::path& path) const; virtual std::vector find_from_PATH(const std::string& name) const = 0; }; Filesystem& get_real_filesystem(); static constexpr const char* FILESYSTEM_INVALID_CHARACTERS = R"(\/:*?"<>|)"; bool has_invalid_chars_for_filesystem(const std::string& s); void print_paths(const std::vector& paths); }