2016-09-19 11:50:08 +08:00
|
|
|
#include "vcpkg_Files.h"
|
|
|
|
#include <fstream>
|
2016-09-24 08:57:18 +08:00
|
|
|
#include <regex>
|
2016-12-01 06:08:43 +08:00
|
|
|
#include "vcpkg_System.h"
|
2016-09-19 11:50:08 +08:00
|
|
|
|
|
|
|
namespace vcpkg {namespace Files
|
|
|
|
{
|
2016-09-24 08:57:18 +08:00
|
|
|
static const std::regex FILESYSTEM_INVALID_CHARACTERS_REGEX = std::regex(R"([\/:*?"<>|])");
|
|
|
|
|
2016-09-19 11:50:08 +08:00
|
|
|
void check_is_directory(const fs::path& dirpath)
|
|
|
|
{
|
2016-12-17 11:37:11 +08:00
|
|
|
Checks::check_exit(fs::is_directory(dirpath), "The path %s is not a directory", dirpath.string());
|
2016-09-19 11:50:08 +08:00
|
|
|
}
|
|
|
|
|
2016-09-24 08:57:18 +08:00
|
|
|
bool has_invalid_chars_for_filesystem(const std::string s)
|
|
|
|
{
|
|
|
|
return std::regex_search(s, FILESYSTEM_INVALID_CHARACTERS_REGEX);
|
|
|
|
}
|
|
|
|
|
2016-09-19 11:50:08 +08:00
|
|
|
expected<std::string> get_contents(const fs::path& file_path) noexcept
|
|
|
|
{
|
|
|
|
std::fstream file_stream(file_path, std::ios_base::in | std::ios_base::binary);
|
|
|
|
if (file_stream.fail())
|
|
|
|
{
|
|
|
|
return std::errc::no_such_file_or_directory;
|
|
|
|
}
|
|
|
|
|
|
|
|
file_stream.seekg(0, file_stream.end);
|
|
|
|
auto length = file_stream.tellg();
|
|
|
|
file_stream.seekg(0, file_stream.beg);
|
|
|
|
|
|
|
|
if (length > SIZE_MAX)
|
|
|
|
{
|
|
|
|
return std::errc::file_too_large;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string output;
|
|
|
|
output.resize(static_cast<size_t>(length));
|
|
|
|
file_stream.read(&output[0], length);
|
|
|
|
file_stream.close();
|
|
|
|
|
|
|
|
return std::move(output);
|
|
|
|
}
|
2016-09-22 12:43:59 +08:00
|
|
|
|
2016-12-16 10:19:22 +08:00
|
|
|
expected<std::vector<std::string>> read_all_lines(const fs::path& file_path)
|
|
|
|
{
|
|
|
|
std::fstream file_stream(file_path, std::ios_base::in | std::ios_base::binary);
|
|
|
|
if (file_stream.fail())
|
|
|
|
{
|
|
|
|
return std::errc::no_such_file_or_directory;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<std::string> output;
|
|
|
|
std::string line;
|
|
|
|
while (std::getline(file_stream, line))
|
|
|
|
{
|
|
|
|
output.push_back(line);
|
|
|
|
}
|
|
|
|
file_stream.close();
|
|
|
|
|
|
|
|
return std::move(output);
|
|
|
|
}
|
|
|
|
|
|
|
|
void write_all_lines(const fs::path& file_path, const std::vector<std::string>& lines)
|
|
|
|
{
|
|
|
|
std::fstream output(file_path, std::ios_base::out | std::ios_base::binary | std::ios_base::trunc);
|
|
|
|
for (const std::string& line : lines)
|
|
|
|
{
|
|
|
|
output << line << "\n";
|
|
|
|
}
|
|
|
|
output.close();
|
|
|
|
}
|
|
|
|
|
2016-09-22 12:43:59 +08:00
|
|
|
fs::path find_file_recursively_up(const fs::path& starting_dir, const std::string& filename)
|
|
|
|
{
|
|
|
|
fs::path current_dir = starting_dir;
|
|
|
|
for (; !current_dir.empty(); current_dir = current_dir.parent_path())
|
|
|
|
{
|
|
|
|
const fs::path candidate = current_dir / filename;
|
|
|
|
if (fs::exists(candidate))
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return current_dir;
|
|
|
|
}
|
2016-11-30 09:04:41 +08:00
|
|
|
|
|
|
|
void recursive_find_files_with_extension_in_dir(const fs::path& dir, const std::string& extension, std::vector<fs::path>* output)
|
|
|
|
{
|
|
|
|
recursive_find_matching_paths_in_dir(dir, [&extension](const fs::path& current)
|
|
|
|
{
|
|
|
|
return !fs::is_directory(current) && current.extension() == extension;
|
|
|
|
}, output);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<fs::path> recursive_find_files_with_extension_in_dir(const fs::path& dir, const std::string& extension)
|
|
|
|
{
|
|
|
|
std::vector<fs::path> v;
|
|
|
|
recursive_find_files_with_extension_in_dir(dir, extension, &v);
|
|
|
|
return v;
|
|
|
|
}
|
2016-11-30 10:08:00 +08:00
|
|
|
|
|
|
|
void recursive_find_all_files_in_dir(const fs::path& dir, std::vector<fs::path>* output)
|
|
|
|
{
|
|
|
|
recursive_find_matching_paths_in_dir(dir, [&](const fs::path& current)
|
2016-11-30 10:08:53 +08:00
|
|
|
{
|
|
|
|
return !fs::is_directory(current);
|
|
|
|
}, output);
|
2016-11-30 10:08:00 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<fs::path> recursive_find_all_files_in_dir(const fs::path& dir)
|
|
|
|
{
|
|
|
|
std::vector<fs::path> v;
|
|
|
|
recursive_find_all_files_in_dir(dir, &v);
|
|
|
|
return v;
|
|
|
|
}
|
|
|
|
|
|
|
|
void non_recursive_find_all_files_in_dir(const fs::path& dir, std::vector<fs::path>* output)
|
|
|
|
{
|
|
|
|
non_recursive_find_matching_paths_in_dir(dir, [&](const fs::path& current)
|
2016-11-30 10:08:53 +08:00
|
|
|
{
|
|
|
|
return !fs::is_directory(current);
|
|
|
|
}, output);
|
2016-11-30 10:08:00 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<fs::path> non_recursive_find_all_files_in_dir(const fs::path& dir)
|
|
|
|
{
|
|
|
|
std::vector<fs::path> v;
|
|
|
|
non_recursive_find_all_files_in_dir(dir, &v);
|
|
|
|
return v;
|
|
|
|
}
|
2016-12-01 06:08:43 +08:00
|
|
|
|
|
|
|
void print_paths(const std::vector<fs::path>& paths)
|
|
|
|
{
|
|
|
|
System::println("");
|
|
|
|
for (const fs::path& p : paths)
|
|
|
|
{
|
|
|
|
System::println(" %s", p.generic_string());
|
|
|
|
}
|
|
|
|
System::println("");
|
|
|
|
}
|
2016-09-19 11:50:08 +08:00
|
|
|
}}
|