#include "platform/platform.hpp" #include "platform/socket.hpp" #include "coding/file_reader.hpp" #include "base/exception.hpp" #include "base/file_name_utils.hpp" #include "base/logging.hpp" #include "base/macros.hpp" #include "base/scope_guard.hpp" #include "std/algorithm.hpp" #include "std/bind.hpp" #include "std/initializer_list.hpp" #include "std/string.hpp" #include #include #include #include #include #include #include namespace { // Web service ip to check internet connection. Now it's a mail.ru ip. char constexpr kSomeWorkingWebServer[] = "217.69.139.202"; // Returns directory where binary resides, including slash at the end. bool GetBinaryDir(string & outPath) { char path[4096] = {}; if (::readlink("/proc/self/exe", path, ARRAY_SIZE(path)) <= 0) return false; 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 namespace platform { unique_ptr CreateSocket() { return unique_ptr(); } } Platform::Platform() { // Init directories. string path; CHECK(GetBinaryDir(path), ("Can't retrieve path to executable")); m_settingsDir = base::JoinPath(HomeDir(), ".config", "MapsWithMe"); if (!IsFileExistsByFullPath(base::JoinPath(m_settingsDir, SETTINGS_FILE_NAME))) { auto const configDir = base::JoinPath(HomeDir(), ".config"); if (!MkDirChecked(configDir)) MYTHROW(FileSystemException, ("Can't create directory", configDir)); if (!MkDirChecked(m_settingsDir)) MYTHROW(FileSystemException, ("Can't create directory", m_settingsDir)); } 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, "..", "..", "..", "omim", "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_settingsDir += '/'; m_writableDir += '/'; char const * tmpDir = ::getenv("TMPDIR"); if (tmpDir) m_tmpDir = tmpDir; else m_tmpDir = "/tmp"; m_tmpDir += '/'; m_privateDir = m_settingsDir; m_guiThread = make_unique(); LOG(LDEBUG, ("Resources directory:", m_resourcesDir)); LOG(LDEBUG, ("Writable directory:", m_writableDir)); LOG(LDEBUG, ("Tmp directory:", m_tmpDir)); LOG(LDEBUG, ("Settings directory:", m_settingsDir)); LOG(LDEBUG, ("Client ID:", UniqueClientId())); } string Platform::UniqueClientId() const { string machineFile = "/var/lib/dbus/machine-id"; if (IsFileExistsByFullPath("/etc/machine-id")) machineFile = "/etc/machine-id"; if (IsFileExistsByFullPath(machineFile)) { string content; FileReader(machineFile).ReadAsString(content); return content.substr(0, 32); } return "n0dbus0n0lsb00000000000000000000"; } string Platform::AdvertisingId() const { return {}; } string Platform::MacAddress(bool md5Decoded) const { // Not implemented. UNUSED_VALUE(md5Decoded); return {}; } string Platform::DeviceName() const { return OMIM_OS_NAME; } string Platform::DeviceModel() const { return {}; } Platform::EConnectionType Platform::ConnectionStatus() { int socketFd = socket(AF_INET, SOCK_STREAM, 0); SCOPE_GUARD(closeSocket, bind(&close, socketFd)); if (socketFd < 0) return EConnectionType::CONNECTION_NONE; struct sockaddr_in addr; memset(&addr, 0, sizeof(addr)); addr.sin_family = AF_INET; addr.sin_port = htons(80); inet_pton(AF_INET, kSomeWorkingWebServer, &addr.sin_addr); if (connect(socketFd, reinterpret_cast(&addr), sizeof(addr)) < 0) return EConnectionType::CONNECTION_NONE; return EConnectionType::CONNECTION_WIFI; } Platform::ChargingStatus Platform::GetChargingStatus() { return Platform::ChargingStatus::Plugged; } uint8_t Platform::GetBatteryLevel() { // This value is always 100 for desktop. return 100; } void Platform::SetGuiThread(unique_ptr guiThread) { m_guiThread = move(guiThread); }