451 lines
11 KiB
C++
451 lines
11 KiB
C++
#include "platform/platform.hpp"
|
|
|
|
#include "coding/file_reader.hpp"
|
|
#include "coding/internal/file_data.hpp"
|
|
#include "coding/writer.hpp"
|
|
|
|
#include "base/file_name_utils.hpp"
|
|
#include "base/logging.hpp"
|
|
#include "base/string_utils.hpp"
|
|
|
|
#include <algorithm>
|
|
#include <random>
|
|
#include <thread>
|
|
|
|
#include <dirent.h>
|
|
#include <sys/stat.h>
|
|
|
|
#include <cerrno>
|
|
|
|
#include <boost/filesystem.hpp>
|
|
#include <boost/range/iterator_range.hpp>
|
|
|
|
#include "platform/constants.hpp"
|
|
|
|
|
|
#ifdef GEOCORE_OS_MAC
|
|
#include <mach-o/dyld.h>
|
|
#endif
|
|
|
|
namespace fs = boost::filesystem;
|
|
using namespace std;
|
|
|
|
namespace
|
|
{
|
|
string RandomString(size_t length)
|
|
{
|
|
static string const kCharset =
|
|
"0123456789"
|
|
"ABCDEFGHIJKLMNOPQRSTUVWXYZ"
|
|
"abcdefghijklmnopqrstuvwxyz";
|
|
random_device rd;
|
|
mt19937 gen(rd());
|
|
uniform_int_distribution<size_t> dis(0, kCharset.size() - 1);
|
|
string str(length, 0);
|
|
generate_n(str.begin(), length, [&]() {
|
|
return kCharset[dis(gen)];
|
|
});
|
|
return str;
|
|
}
|
|
|
|
bool IsSpecialDirName(string const & dirName)
|
|
{
|
|
return dirName == "." || dirName == "..";
|
|
}
|
|
|
|
bool GetFileTypeChecked(string const & path, Platform::EFileType & type)
|
|
{
|
|
Platform::EError const ret = Platform::GetFileType(path, type);
|
|
if (ret != Platform::ERR_OK)
|
|
{
|
|
LOG(LERROR, ("Can't determine file type for", path, ":", ret));
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
} // namespace
|
|
|
|
// static
|
|
Platform::EError Platform::ErrnoToError()
|
|
{
|
|
switch (errno)
|
|
{
|
|
case ENOENT:
|
|
return ERR_FILE_DOES_NOT_EXIST;
|
|
case EACCES:
|
|
return ERR_ACCESS_FAILED;
|
|
case ENOTEMPTY:
|
|
return ERR_DIRECTORY_NOT_EMPTY;
|
|
case EEXIST:
|
|
return ERR_FILE_ALREADY_EXISTS;
|
|
case ENAMETOOLONG:
|
|
return ERR_NAME_TOO_LONG;
|
|
case ENOTDIR:
|
|
return ERR_NOT_A_DIRECTORY;
|
|
case ELOOP:
|
|
return ERR_SYMLINK_LOOP;
|
|
case EIO:
|
|
return ERR_IO_ERROR;
|
|
default:
|
|
return ERR_UNKNOWN;
|
|
}
|
|
}
|
|
|
|
// static
|
|
bool Platform::RmDirRecursively(string const & dirName)
|
|
{
|
|
if (dirName.empty() || IsSpecialDirName(dirName))
|
|
return false;
|
|
|
|
bool res = true;
|
|
|
|
FilesList allFiles;
|
|
GetFilesByRegExp(dirName, ".*", allFiles);
|
|
for (string const & file : allFiles)
|
|
{
|
|
string const path = base::JoinPath(dirName, file);
|
|
|
|
EFileType type;
|
|
if (GetFileType(path, type) != ERR_OK)
|
|
continue;
|
|
|
|
if (type == FILE_TYPE_DIRECTORY)
|
|
{
|
|
if (!IsSpecialDirName(file) && !RmDirRecursively(path))
|
|
res = false;
|
|
}
|
|
else
|
|
{
|
|
if (!base::DeleteFileX(path))
|
|
res = false;
|
|
}
|
|
}
|
|
|
|
if (RmDir(dirName) != ERR_OK)
|
|
res = false;
|
|
|
|
return res;
|
|
}
|
|
|
|
string Platform::ReadPathForFile(string const & file, string searchScope) const
|
|
{
|
|
if (searchScope.empty())
|
|
searchScope = "wrf";
|
|
|
|
string fullPath;
|
|
for (size_t i = 0; i < searchScope.size(); ++i)
|
|
{
|
|
switch (searchScope[i])
|
|
{
|
|
case 'w': fullPath = m_writableDir + file; break;
|
|
case 'r': fullPath = m_resourcesDir + file; break;
|
|
case 'f': fullPath = file; break;
|
|
default : CHECK(false, ("Unsupported searchScope:", searchScope)); break;
|
|
}
|
|
if (IsFileExistsByFullPath(fullPath))
|
|
return fullPath;
|
|
}
|
|
|
|
string const possiblePaths = m_writableDir + "\n" + m_resourcesDir + "\n";
|
|
MYTHROW(FileAbsentException, ("File", file, "doesn't exist in the scope", searchScope,
|
|
"Have been looking in:\n", possiblePaths));
|
|
}
|
|
|
|
bool Platform::RemoveFileIfExists(string const & filePath)
|
|
{
|
|
return IsFileExistsByFullPath(filePath) ? base::DeleteFileX(filePath) : true;
|
|
}
|
|
|
|
string Platform::TmpPathForFile() const
|
|
{
|
|
size_t const kNameLen = 32;
|
|
return TmpDir() + RandomString(kNameLen);
|
|
}
|
|
|
|
void Platform::GetFilesByExt(string const & directory, string const & ext, FilesList & outFiles)
|
|
{
|
|
// Transform extension mask to regexp (.mwm -> \.mwm$)
|
|
ASSERT ( !ext.empty(), () );
|
|
ASSERT_EQUAL ( ext[0], '.' , () );
|
|
|
|
GetFilesByRegExp(directory, '\\' + ext + '$', outFiles);
|
|
}
|
|
|
|
// static
|
|
void Platform::GetFilesByType(string const & directory, unsigned typeMask,
|
|
TFilesWithType & outFiles)
|
|
{
|
|
FilesList allFiles;
|
|
GetFilesByRegExp(directory, ".*", allFiles);
|
|
for (string const & file : allFiles)
|
|
{
|
|
EFileType type;
|
|
if (GetFileType(base::JoinPath(directory, file), type) != ERR_OK)
|
|
continue;
|
|
if (typeMask & type)
|
|
outFiles.emplace_back(file, type);
|
|
}
|
|
}
|
|
|
|
// static
|
|
bool Platform::IsDirectory(string const & path)
|
|
{
|
|
EFileType fileType;
|
|
if (GetFileType(path, fileType) != ERR_OK)
|
|
return false;
|
|
return fileType == FILE_TYPE_DIRECTORY;
|
|
}
|
|
|
|
// static
|
|
void Platform::GetFilesRecursively(string const & directory, FilesList & filesList)
|
|
{
|
|
TFilesWithType files;
|
|
|
|
GetFilesByType(directory, Platform::FILE_TYPE_REGULAR, files);
|
|
for (auto const & p : files)
|
|
{
|
|
auto const & file = p.first;
|
|
CHECK_EQUAL(p.second, Platform::FILE_TYPE_REGULAR, ("dir:", directory, "file:", file));
|
|
filesList.push_back(base::JoinPath(directory, file));
|
|
}
|
|
|
|
TFilesWithType subdirs;
|
|
GetFilesByType(directory, Platform::FILE_TYPE_DIRECTORY, subdirs);
|
|
|
|
for (auto const & p : subdirs)
|
|
{
|
|
auto const & subdir = p.first;
|
|
CHECK_EQUAL(p.second, Platform::FILE_TYPE_DIRECTORY, ("dir:", directory, "subdir:", subdir));
|
|
if (subdir == "." || subdir == "..")
|
|
continue;
|
|
|
|
GetFilesRecursively(base::JoinPath(directory, subdir), filesList);
|
|
}
|
|
}
|
|
|
|
void Platform::SetWritableDir(string const & path)
|
|
{
|
|
m_writableDir = base::AddSlashIfNeeded(path);
|
|
}
|
|
|
|
void Platform::SetResourceDir(string const & path)
|
|
{
|
|
m_resourcesDir = base::AddSlashIfNeeded(path);
|
|
}
|
|
|
|
// static
|
|
bool Platform::MkDirChecked(string const & dirName)
|
|
{
|
|
Platform::EError const ret = MkDir(dirName);
|
|
switch (ret)
|
|
{
|
|
case Platform::ERR_OK: return true;
|
|
case Platform::ERR_FILE_ALREADY_EXISTS:
|
|
{
|
|
Platform::EFileType type;
|
|
if (!GetFileTypeChecked(dirName, type))
|
|
return false;
|
|
if (type != Platform::FILE_TYPE_DIRECTORY)
|
|
{
|
|
LOG(LERROR, (dirName, "exists, but not a dirName:", type));
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
default: LOG(LERROR, (dirName, "can't be created:", ret)); return false;
|
|
}
|
|
}
|
|
|
|
// static
|
|
bool Platform::MkDirRecursively(string const & dirName)
|
|
{
|
|
auto const sep = base::GetNativeSeparator();
|
|
string path = strings::StartsWith(dirName, sep) ? sep : "";
|
|
auto const tokens = strings::Tokenize(dirName, sep.c_str());
|
|
for (auto const & t : tokens)
|
|
{
|
|
path = base::JoinPath(path, t);
|
|
if (!IsFileExistsByFullPath(path))
|
|
{
|
|
auto const ret = MkDir(path);
|
|
switch (ret)
|
|
{
|
|
case ERR_OK: break;
|
|
case ERR_FILE_ALREADY_EXISTS:
|
|
{
|
|
if (!IsDirectory(path))
|
|
return false;
|
|
break;
|
|
}
|
|
default: return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
// static
|
|
unsigned Platform::CpuCores() const
|
|
{
|
|
unsigned const cores = thread::hardware_concurrency();
|
|
return cores > 0 ? cores : 1;
|
|
}
|
|
|
|
// static
|
|
Platform::EError Platform::RmDir(string const & dirName)
|
|
{
|
|
if (rmdir(dirName.c_str()) != 0)
|
|
return ErrnoToError();
|
|
return ERR_OK;
|
|
}
|
|
|
|
// static
|
|
Platform::EError Platform::GetFileType(string const & path, EFileType & type)
|
|
{
|
|
struct stat stats;
|
|
if (stat(path.c_str(), &stats) != 0)
|
|
return ErrnoToError();
|
|
if (S_ISREG(stats.st_mode))
|
|
type = FILE_TYPE_REGULAR;
|
|
else if (S_ISDIR(stats.st_mode))
|
|
type = FILE_TYPE_DIRECTORY;
|
|
else
|
|
type = FILE_TYPE_UNKNOWN;
|
|
return ERR_OK;
|
|
}
|
|
|
|
// static
|
|
bool Platform::IsFileExistsByFullPath(string const & filePath)
|
|
{
|
|
struct stat s;
|
|
return stat(filePath.c_str(), &s) == 0;
|
|
}
|
|
|
|
// static
|
|
string Platform::GetCurrentWorkingDirectory() noexcept
|
|
{
|
|
char path[PATH_MAX];
|
|
char const * const dir = getcwd(path, PATH_MAX);
|
|
if (dir == nullptr)
|
|
return {};
|
|
return dir;
|
|
}
|
|
|
|
bool Platform::IsDirectoryEmpty(string const & directory)
|
|
{
|
|
auto closeMe = [](DIR * dir){ closedir(dir) };
|
|
unique_ptr<DIR, decltype(closeMe) }> dir(opendir(directory.c_str()));
|
|
if (!dir)
|
|
return true;
|
|
|
|
struct dirent * entry;
|
|
|
|
// Invariant: all files met so far are "." or "..".
|
|
while ((entry = readdir(dir.get())) != nullptr)
|
|
{
|
|
// A file is not a special UNIX file. Early exit here.
|
|
if (strcmp(entry->d_name, ".") != 0 && strcmp(entry->d_name, "..") != 0)
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool Platform::GetFileSizeByFullPath(string const & filePath, uint64_t & size)
|
|
{
|
|
struct stat s;
|
|
if (stat(filePath.c_str(), &s) == 0)
|
|
{
|
|
size = s.st_size;
|
|
return true;
|
|
}
|
|
else return false;
|
|
}
|
|
|
|
Platform::Platform()
|
|
{
|
|
char const * tmpDir = ::getenv("TMPDIR");
|
|
if (tmpDir)
|
|
m_tmpDir = tmpDir;
|
|
else
|
|
m_tmpDir = "/tmp";
|
|
|
|
m_tmpDir += '/';
|
|
|
|
LOG(LDEBUG, ("Tmp directory:", m_tmpDir));
|
|
}
|
|
|
|
unique_ptr<ModelReader> Platform::GetReader(string const & file, string const & searchScope) const
|
|
{
|
|
return make_unique<FileReader>(ReadPathForFile(file, searchScope), READER_CHUNK_LOG_SIZE,
|
|
READER_CHUNK_LOG_COUNT);
|
|
}
|
|
|
|
bool Platform::GetFileSizeByName(string const & fileName, uint64_t & size) const
|
|
{
|
|
try
|
|
{
|
|
return GetFileSizeByFullPath(ReadPathForFile(fileName), size);
|
|
}
|
|
catch (RootException const &)
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
void Platform::GetFilesByRegExp(string const & directory, string const & regexp,
|
|
FilesList & outFiles)
|
|
{
|
|
boost::system::error_code ec{};
|
|
regex exp(regexp);
|
|
|
|
for (auto const & entry : boost::make_iterator_range(fs::directory_iterator(directory, ec), {}))
|
|
{
|
|
string const name = entry.path().filename().string();
|
|
if (regex_search(name.begin(), name.end(), exp))
|
|
outFiles.push_back(name);
|
|
}
|
|
}
|
|
|
|
// static
|
|
Platform::EError Platform::MkDir(string const & dirName)
|
|
{
|
|
boost::system::error_code ec{};
|
|
|
|
fs::path dirPath{dirName};
|
|
if (fs::exists(dirPath, ec))
|
|
return Platform::ERR_FILE_ALREADY_EXISTS;
|
|
if (!fs::create_directory(dirPath, ec))
|
|
{
|
|
LOG(LWARNING, ("Can't create directory: ", dirName));
|
|
return Platform::ERR_UNKNOWN;
|
|
}
|
|
return Platform::ERR_OK;
|
|
}
|
|
|
|
Platform & GetPlatform()
|
|
{
|
|
static Platform platform;
|
|
return platform;
|
|
}
|
|
|
|
string DebugPrint(Platform::EError err)
|
|
{
|
|
switch (err)
|
|
{
|
|
case Platform::ERR_OK: return "Ok";
|
|
case Platform::ERR_FILE_DOES_NOT_EXIST: return "File does not exist.";
|
|
case Platform::ERR_ACCESS_FAILED: return "Access failed.";
|
|
case Platform::ERR_DIRECTORY_NOT_EMPTY: return "Directory not empty.";
|
|
case Platform::ERR_FILE_ALREADY_EXISTS: return "File already exists.";
|
|
case Platform::ERR_NAME_TOO_LONG:
|
|
return "The length of a component of path exceeds {NAME_MAX} characters.";
|
|
case Platform::ERR_NOT_A_DIRECTORY:
|
|
return "A component of the path prefix of Path is not a directory.";
|
|
case Platform::ERR_SYMLINK_LOOP:
|
|
return "Too many symbolic links were encountered in translating path.";
|
|
case Platform::ERR_IO_ERROR: return "An I/O error occurred.";
|
|
case Platform::ERR_UNKNOWN: return "Unknown";
|
|
}
|
|
UNREACHABLE();
|
|
}
|