diff --git a/generator/generator_tool/generator_tool.cpp b/generator/generator_tool/generator_tool.cpp index 5905e95..5ec5163 100644 --- a/generator/generator_tool/generator_tool.cpp +++ b/generator/generator_tool/generator_tool.cpp @@ -212,16 +212,22 @@ int GeneratorToolMain(int argc, char ** argv) Platform & pl = GetPlatform(); auto threadsCount = pl.CpuCores(); - if (!options.m_user_resource_path.empty()) + if (options.m_user_resource_path.empty()) { - pl.SetResourceDir(options.m_user_resource_path); + LOG(LERROR, ("Set user resource path")); + return EXIT_FAILURE; } - string const path = - options.m_data_path.empty() ? pl.WritableDir() : base::AddSlashIfNeeded(options.m_data_path); + if (options.m_data_path.empty()) + { + LOG(LERROR, ("Set data path path")); + return EXIT_FAILURE; + } - // So that stray GetWritablePathForFile calls do not crash the generator. - pl.SetWritableDirForTests(path); + pl.SetWritableDir(options.m_data_path); + pl.SetResourceDir(options.m_user_resource_path); + + string const path = base::AddSlashIfNeeded(options.m_data_path); feature::GenerateInfo genInfo; genInfo.m_verbose = options.m_verbose; diff --git a/platform/platform.cpp b/platform/platform.cpp index 2b784b4..88f4b74 100644 --- a/platform/platform.cpp +++ b/platform/platform.cpp @@ -223,7 +223,7 @@ void Platform::GetFilesRecursively(string const & directory, FilesList & filesLi } } -void Platform::SetWritableDirForTests(string const & path) +void Platform::SetWritableDir(string const & path) { m_writableDir = base::AddSlashIfNeeded(path); } @@ -285,24 +285,13 @@ bool Platform::MkDirRecursively(string const & dirName) return true; } +// static unsigned Platform::CpuCores() const { unsigned const cores = thread::hardware_concurrency(); return cores > 0 ? cores : 1; } -namespace -{ -struct CloseDir -{ - void operator()(DIR * dir) const - { - if (dir) - closedir(dir); - } -}; -} // namespace - // static Platform::EError Platform::RmDir(string const & dirName) { @@ -345,7 +334,8 @@ string Platform::GetCurrentWorkingDirectory() noexcept bool Platform::IsDirectoryEmpty(string const & directory) { - unique_ptr dir(opendir(directory.c_str())); + auto closeMe = [](DIR * dir){ closedir(dir) }; + unique_ptr dir(opendir(directory.c_str())); if (!dir) return true; @@ -372,165 +362,19 @@ bool Platform::GetFileSizeByFullPath(string const & filePath, uint64_t & size) else return false; } -namespace -{ - -// Returns directory where binary resides, including slash at the end. -bool GetBinaryDir(string & outPath) -{ - char path[4096] = {}; -#ifdef GEOCORE_OS_MAC - uint32_t size = 4096; - if (_NSGetExecutablePath(path, &size) != 0) - return false; -#else - if (::readlink("/proc/self/exe", path, ARRAY_SIZE(path)) <= 0) - return false; -#endif - outPath = path; - outPath.erase(outPath.find_last_of('/') + 1); - return true; -} - -// Returns true if EULA file exists in directory. -bool IsEulaExist(string const & directory) -{ - return Platform::IsFileExistsByFullPath(base::JoinPath(directory, "eula.html")); -} - -// Makes base::JoinPath(path, dirs) and all intermediate dirs. -// The directory |path| is assumed to exist already. -bool MkDirsChecked(string path, initializer_list const & dirs) -{ - string accumulatedDirs = path; - // Storing full paths is redundant but makes the implementation easier. - vector madeDirs; - bool ok = true; - for (auto const & dir : dirs) - { - accumulatedDirs = base::JoinPath(accumulatedDirs, dir); - auto const result = Platform::MkDir(accumulatedDirs); - switch (result) - { - case Platform::ERR_OK: madeDirs.push_back(accumulatedDirs); break; - case Platform::ERR_FILE_ALREADY_EXISTS: - { - Platform::EFileType type; - if (Platform::GetFileType(accumulatedDirs, type) != Platform::ERR_OK || - type != Platform::FILE_TYPE_DIRECTORY) - { - ok = false; - } - } - break; - default: ok = false; break; - } - } - - if (ok) - return true; - - for (; !madeDirs.empty(); madeDirs.pop_back()) - Platform::RmDir(madeDirs.back()); - - return false; -} - -string HomeDir() -{ - char const * homePath = ::getenv("HOME"); - if (homePath == nullptr) - MYTHROW(RootException, ("The environment variable HOME is not set")); - return homePath; -} - -// Returns the default path to the writable dir, creating the dir if needed. -// An exception is thrown if the default dir is not already there and we were unable to create it. -string DefaultWritableDir() -{ - initializer_list const dirs = {".local", "share", "MapsWithMe"}; - string result; - for (auto const & dir : dirs) - result = base::JoinPath(result, dir); - result = base::AddSlashIfNeeded(result); - - auto const home = HomeDir(); - if (!MkDirsChecked(home, dirs)) - MYTHROW(FileSystemException, ("Cannot create directory:", result)); - return result; -} -} // namespace - - Platform::Platform() { - // Init directories. - string path; - CHECK(GetBinaryDir(path), ("Can't retrieve path to executable")); - - - char const * resDir = ::getenv("MWM_RESOURCES_DIR"); - char const * writableDir = ::getenv("MWM_WRITABLE_DIR"); - if (resDir && writableDir) - { - m_resourcesDir = resDir; - m_writableDir = writableDir; - } - else if (resDir) - { - m_resourcesDir = resDir; - m_writableDir = DefaultWritableDir(); - } - else - { - string const devBuildWithSymlink = base::JoinPath(path, "..", "..", "data"); - string const devBuildWithoutSymlink = base::JoinPath(path, "..", "..", "..", "geocore", "data"); - string const installedVersionWithPackages = base::JoinPath(path, "..", "share"); - string const installedVersionWithoutPackages = base::JoinPath(path, "..", "MapsWithMe"); - string const customInstall = path; - - if (IsEulaExist(devBuildWithSymlink)) - { - m_resourcesDir = devBuildWithSymlink; - m_writableDir = writableDir != nullptr ? writableDir : m_resourcesDir; - } - else if (IsEulaExist(devBuildWithoutSymlink)) - { - m_resourcesDir = devBuildWithoutSymlink; - m_writableDir = writableDir != nullptr ? writableDir : m_resourcesDir; - } - else if (IsEulaExist(installedVersionWithPackages)) - { - m_resourcesDir = installedVersionWithPackages; - m_writableDir = writableDir != nullptr ? writableDir : DefaultWritableDir(); - } - else if (IsEulaExist(installedVersionWithoutPackages)) - { - m_resourcesDir = installedVersionWithoutPackages; - m_writableDir = writableDir != nullptr ? writableDir : DefaultWritableDir(); - } - else if (IsEulaExist(customInstall)) - { - m_resourcesDir = path; - m_writableDir = writableDir != nullptr ? writableDir : DefaultWritableDir(); - } - } - m_resourcesDir += '/'; - m_writableDir += '/'; - char const * tmpDir = ::getenv("TMPDIR"); if (tmpDir) m_tmpDir = tmpDir; else m_tmpDir = "/tmp"; + m_tmpDir += '/'; - LOG(LDEBUG, ("Resources directory:", m_resourcesDir)); - LOG(LDEBUG, ("Writable directory:", m_writableDir)); LOG(LDEBUG, ("Tmp directory:", m_tmpDir)); } - unique_ptr Platform::GetReader(string const & file, string const & searchScope) const { return make_unique(ReadPathForFile(file, searchScope), READER_CHUNK_LOG_SIZE, diff --git a/platform/platform.hpp b/platform/platform.hpp index 11c3fd2..58460d8 100644 --- a/platform/platform.hpp +++ b/platform/platform.hpp @@ -53,12 +53,12 @@ public: protected: /// Usually read-only directory for application resources - std::string m_resourcesDir; + std::string m_resourcesDir = "./data"; /// Writable directory to store downloaded map data /// @note on some systems it can point to external ejectable storage - std::string m_writableDir; + std::string m_writableDir = "."; /// Temporary directory, can be cleaned up by the system - std::string m_tmpDir; + std::string m_tmpDir = "/tmp"; /// Returns last system call error as EError. static EError ErrnoToError(); @@ -76,8 +76,8 @@ public: static std::string GetCurrentWorkingDirectory() noexcept; /// @return always the same writable dir for current user with slash at the end std::string const & WritableDir() const { return m_writableDir; } - /// Set writable dir — use for testing and linux stuff only - void SetWritableDirForTests(std::string const & path); + /// Set writable dir + void SetWritableDir(std::string const & path); /// @return full path to file in user's writable directory std::string WritablePathForFile(std::string const & file) const { return WritableDir() + file; } /// Uses m_writeableDir [w], m_resourcesDir [r] @@ -161,7 +161,7 @@ public: // Please note, that number of active cores can vary at runtime. // DO NOT assume for the same return value between calls. - unsigned CpuCores() const; + static unsigned CpuCores(); }; std::string DebugPrint(Platform::EError err); diff --git a/testing/testingmain.cpp b/testing/testingmain.cpp index 13e201f..7e392c4 100644 --- a/testing/testingmain.cpp +++ b/testing/testingmain.cpp @@ -10,7 +10,7 @@ int main(int argc, char * argv[]) #ifndef GEOCORE_UNIT_TEST_DISABLE_PLATFORM_INIT // Setting stored paths from testingmain.cpp Platform & pl = GetPlatform(); - pl.SetWritableDirForTests(TestindDataPath::kDataPath); + pl.SetWritableDir(TestindDataPath::kDataPath); pl.SetResourceDir(TestindDataPath::kDataPath); #endif