From feeff70bd7529a44ac676a4e431dde2f9214ec63 Mon Sep 17 00:00:00 2001 From: Viktor Govako Date: Mon, 18 Apr 2022 20:51:33 +0300 Subject: [PATCH] [android] Added World files into countries list. Signed-off-by: Viktor Govako --- .../maps/DownloadResourcesLegacyActivity.cpp | 99 ++++-------- android/jni/com/mapswithme/maps/Framework.cpp | 17 +- android/jni/com/mapswithme/maps/Framework.hpp | 1 + .../maps/DownloadResourcesLegacyActivity.java | 7 +- .../src/com/mapswithme/maps/Framework.java | 4 +- .../maps/downloader/DownloaderAdapter.java | 16 +- data/countries.txt | 10 ++ platform/local_country_file.hpp | 1 + platform/local_country_file_utils.cpp | 51 +++--- storage/country_info_getter.cpp | 13 +- storage/storage.cpp | 146 +++++++++++++++--- storage/storage.hpp | 10 +- storage/storage_tests/CMakeLists.txt | 1 + storage/storage_tests/countries_tests.cpp | 14 ++ storage/storage_tests/storage_tests.cpp | 15 +- 15 files changed, 248 insertions(+), 157 deletions(-) create mode 100644 storage/storage_tests/countries_tests.cpp diff --git a/android/jni/com/mapswithme/maps/DownloadResourcesLegacyActivity.cpp b/android/jni/com/mapswithme/maps/DownloadResourcesLegacyActivity.cpp index e577336829..50834befd2 100644 --- a/android/jni/com/mapswithme/maps/DownloadResourcesLegacyActivity.cpp +++ b/android/jni/com/mapswithme/maps/DownloadResourcesLegacyActivity.cpp @@ -40,18 +40,10 @@ using namespace std::placeholders; #define ERR_FILE_IN_PROGRESS -6 //@} -struct FileToDownload -{ - std::vector m_urls; - std::string m_fileName; - std::string m_pathOnSdcard; - uint64_t m_fileSize; -}; - namespace { -static std::vector g_filesToDownload; +static std::vector g_filesToDownload; static int g_totalDownloadedBytes; static int g_totalBytesToDownload; static std::shared_ptr g_currentRequest; @@ -77,24 +69,6 @@ extern "C" } } - // Check if we need to download mandatory resource file. - static bool NeedToDownload(Platform & pl, std::string const & name, int size) - { - try - { - ModelReaderPtr reader(pl.GetReader(name)); - return false; - } - catch (RootException const &) - { - // TODO: A hack for a changed font file name. Remove when this stops being relevant. - // This is because we don't want to force update users with the old font (which is good enough) - if (name == "02_droidsans-fallback.ttf") - return NeedToDownload(pl, "02_wqy-microhei.ttf", size); // Size is not checked anyway - } - return true; - } - JNIEXPORT jint JNICALL Java_com_mapswithme_maps_DownloadResourcesLegacyActivity_nativeGetBytesToDownload(JNIEnv * env, jclass clazz) { @@ -103,39 +77,14 @@ extern "C" g_totalBytesToDownload = 0; g_totalDownloadedBytes = 0; + storage::Storage const & storage = g_framework->GetStorage(); + bool const anyWorldWasMoved = storage.GetForceDownloadWorlds(g_filesToDownload); + + for (auto const & cf : g_filesToDownload) + g_totalBytesToDownload += cf.GetRemoteSize(); + Platform & pl = GetPlatform(); - std::string const path = pl.WritableDir(); - - ReaderStreamBuf buffer(pl.GetReader(EXTERNAL_RESOURCES_FILE)); - std::istream in(&buffer); - - std::string name; - int size; - while (true) - { - in >> name; - if (!in.good()) - break; - - in >> size; - if (!in.good()) - break; - - if (NeedToDownload(pl, name, size)) - { - LOG(LDEBUG, ("Should download", name, "size", size, "bytes")); - - FileToDownload f; - f.m_pathOnSdcard = path + name; - f.m_fileName = name; - f.m_fileSize = size; - - g_filesToDownload.push_back(f); - g_totalBytesToDownload += size; - } - } - - int const res = HasSpaceForFiles(pl, path, g_totalBytesToDownload); + int const res = HasSpaceForFiles(pl, pl.WritableDir(), g_totalBytesToDownload); if (res == ERR_STORAGE_DISCONNECTED) LOG(LWARNING, ("External file system is not available")); else if (res == ERR_NOT_ENOUGH_FREE_SPACE) @@ -143,6 +92,9 @@ extern "C" g_currentRequest.reset(); + if (anyWorldWasMoved && res == 0) + g_framework->ReloadWorldMaps(); + return res; } @@ -159,10 +111,11 @@ extern "C" if (errorCode == ERR_DOWNLOAD_SUCCESS) { - FileToDownload & curFile = g_filesToDownload.back(); - LOG(LDEBUG, ("finished downloading", curFile.m_fileName, "size", curFile.m_fileSize, "bytes")); + auto const & curFile = g_filesToDownload.back(); + size_t const sz = curFile.GetRemoteSize(); + LOG(LDEBUG, ("finished downloading", curFile.GetName(), "size", sz, "bytes")); - g_totalDownloadedBytes += curFile.m_fileSize; + g_totalDownloadedBytes += sz; LOG(LDEBUG, ("totalDownloadedBytes:", g_totalDownloadedBytes)); g_filesToDownload.pop_back(); @@ -189,18 +142,22 @@ extern "C" /// @todo One downloader instance with cached servers. All this routine will be refactored some time. static auto downloader = storage::GetDownloader(); - downloader->SetDataVersion(g_framework->GetStorage().GetCurrentDataVersion()); + storage::Storage const & storage = g_framework->GetStorage(); + downloader->SetDataVersion(storage.GetCurrentDataVersion()); - downloader->EnsureServersListReady([ptr = jni::make_global_ref(listener)]() + downloader->EnsureServersListReady([&storage, ptr = jni::make_global_ref(listener)]() { - FileToDownload const & curFile = g_filesToDownload.back(); - LOG(LINFO, ("Downloading file", curFile.m_fileName)); + auto const & curFile = g_filesToDownload.back(); + auto const fileName = curFile.GetFileName(MapFileType::Map); + LOG(LINFO, ("Downloading file", fileName)); - auto const urls = downloader->MakeUrlListLegacy(curFile.m_fileName); - g_currentRequest.reset(HttpRequest::GetFile(urls, curFile.m_pathOnSdcard, curFile.m_fileSize, - std::bind(&DownloadFileFinished, ptr, _1), - std::bind(&DownloadFileProgress, ptr, _1), - 512 * 1024, false)); + g_currentRequest.reset(HttpRequest::GetFile( + downloader->MakeUrlListLegacy(fileName), + storage.GetFilePath(curFile.GetName(), MapFileType::Map), + curFile.GetRemoteSize(), + std::bind(&DownloadFileFinished, ptr, _1), + std::bind(&DownloadFileProgress, ptr, _1), + 512 * 1024, false)); }); return ERR_FILE_IN_PROGRESS; diff --git a/android/jni/com/mapswithme/maps/Framework.cpp b/android/jni/com/mapswithme/maps/Framework.cpp index 88106faeec..5563e46a8b 100644 --- a/android/jni/com/mapswithme/maps/Framework.cpp +++ b/android/jni/com/mapswithme/maps/Framework.cpp @@ -554,6 +554,13 @@ void Framework::RemoveLocalMaps() m_work.DeregisterAllMaps(); } +void Framework::ReloadWorldMaps() +{ + /// @todo Can invent more optimal routine to remove/add World files only. + RemoveLocalMaps(); + AddLocalMaps(); +} + void Framework::ReplaceBookmark(kml::MarkId markId, kml::BookmarkData & bm) { m_work.GetBookmarkManager().GetEditSession().UpdateBookmark(markId, bm); @@ -1506,15 +1513,9 @@ Java_com_mapswithme_maps_Framework_nativeGetTransitRouteInfo(JNIEnv * env, jclas } JNIEXPORT void JNICALL -Java_com_mapswithme_maps_Framework_nativeRegisterMaps(JNIEnv * env, jclass) +Java_com_mapswithme_maps_Framework_nativeReloadWorldMaps(JNIEnv * env, jclass) { - frm()->RegisterAllMaps(); -} - -JNIEXPORT void JNICALL -Java_com_mapswithme_maps_Framework_nativeDeregisterMaps(JNIEnv * env, jclass) -{ - frm()->DeregisterAllMaps(); + g_framework->ReloadWorldMaps(); } JNIEXPORT jboolean JNICALL diff --git a/android/jni/com/mapswithme/maps/Framework.hpp b/android/jni/com/mapswithme/maps/Framework.hpp index 3950059149..ad693a0632 100644 --- a/android/jni/com/mapswithme/maps/Framework.hpp +++ b/android/jni/com/mapswithme/maps/Framework.hpp @@ -137,6 +137,7 @@ namespace android void AddLocalMaps(); void RemoveLocalMaps(); + void ReloadWorldMaps(); m2::PointD GetViewportCenter() const; diff --git a/android/src/com/mapswithme/maps/DownloadResourcesLegacyActivity.java b/android/src/com/mapswithme/maps/DownloadResourcesLegacyActivity.java index 0ca4a47379..37bf6c3309 100644 --- a/android/src/com/mapswithme/maps/DownloadResourcesLegacyActivity.java +++ b/android/src/com/mapswithme/maps/DownloadResourcesLegacyActivity.java @@ -404,10 +404,9 @@ public class DownloadResourcesLegacyActivity extends BaseMwmFragmentActivity imp { if (result == ERR_NO_MORE_FILES) { - // World and WorldCoasts has been downloaded, we should register maps again to correctly add them to the model and generate indexes etc. - // TODO fix the hack when separate download of World-s will be removed or refactored - Framework.nativeDeregisterMaps(); - Framework.nativeRegisterMaps(); + // World and WorldCoasts has been downloaded, we should register maps again to correctly add them to the model. + Framework.nativeReloadWorldMaps(); + if (mCurrentCountry != null && mChbDownloadCountry.isChecked()) { CountryItem item = CountryItem.fill(mCurrentCountry); diff --git a/android/src/com/mapswithme/maps/Framework.java b/android/src/com/mapswithme/maps/Framework.java index 06e77c4989..fba06cc87c 100644 --- a/android/src/com/mapswithme/maps/Framework.java +++ b/android/src/com/mapswithme/maps/Framework.java @@ -308,9 +308,7 @@ public class Framework /** * Registers all maps(.mwms). Adds them to the models, generates indexes and does all necessary stuff. */ - public static native void nativeRegisterMaps(); - - public static native void nativeDeregisterMaps(); + public static native void nativeReloadWorldMaps(); /** * Determines if currently is day or night at the given location. Used to switch day/night styles. diff --git a/android/src/com/mapswithme/maps/downloader/DownloaderAdapter.java b/android/src/com/mapswithme/maps/downloader/DownloaderAdapter.java index 3af7834645..9db1a5d593 100644 --- a/android/src/com/mapswithme/maps/downloader/DownloaderAdapter.java +++ b/android/src/com/mapswithme/maps/downloader/DownloaderAdapter.java @@ -334,10 +334,14 @@ class DownloaderAdapter extends RecyclerView.Adapter onExploreActionSelected(mItem, DownloaderAdapter.this)); } - private MenuBottomSheetItem getDeleteMenuItem() + private void appendDeleteMenuItem(ArrayList items) { - return new MenuBottomSheetItem(R.string.delete, R.drawable.ic_delete, - () -> onDeleteActionSelected(mItem, DownloaderAdapter.this)); + // Do not show "Delete" option for World files. + // Checking name is not beautiful, but the simplest way for now .. + if (!mItem.id.startsWith("World")) { + items.add(new MenuBottomSheetItem(R.string.delete, R.drawable.ic_delete, + () -> onDeleteActionSelected(mItem, DownloaderAdapter.this))); + } } private MenuBottomSheetItem getCancelMenuItem() @@ -362,7 +366,7 @@ class DownloaderAdapter extends RecyclerView.AdapterGetCountryFile().GetName() == file) + if (i->GetCountryName() == file) break; } try { - Platform & platform = GetPlatform(); - ModelReaderPtr reader( - platform.GetReader(file + DATA_FILE_EXTENSION, GetSpecialFilesSearchScope())); + ModelReaderPtr reader(platform.GetReader(file + DATA_FILE_EXTENSION, GetAdditionalWorldScope())); - // Assume that empty path means the resource file. - LocalCountryFile worldFile{string(), CountryFile(file), version::ReadVersionDate(reader)}; - worldFile.m_files[base::Underlying(MapFileType::Map)] = 1; + // Empty path means the resource file. + LocalCountryFile worldFile(string(), CountryFile(file), version::ReadVersionDate(reader)); + worldFile.m_files[base::Underlying(MapFileType::Map)] = reader.Size(); + + // Replace if newer only. if (i != localFiles.end()) { - // Always use resource World files instead of local on disk. - *i = worldFile; + if (worldFile.GetVersion() > i->GetVersion()) + *i = std::move(worldFile); } else - localFiles.push_back(worldFile); + localFiles.push_back(std::move(worldFile)); } catch (RootException const & ex) { @@ -329,16 +327,13 @@ string GetFileDownloadPath(int64_t version, string const & dataDir, string const return GetFilePath(version, dataDir, countryName, type) + READY_FILE_EXTENSION; } -unique_ptr GetCountryReader(platform::LocalCountryFile const & file, MapFileType type) +unique_ptr GetCountryReader(LocalCountryFile const & file, MapFileType type) { Platform & platform = GetPlatform(); - // See LocalCountryFile comment for explanation. - if (file.GetDirectory().empty()) - { - return platform.GetReader(file.GetCountryName() + DATA_FILE_EXTENSION, - GetSpecialFilesSearchScope()); - } - return platform.GetReader(file.GetPath(type), "f"); + if (file.IsInBundle()) + return platform.GetReader(file.GetFileName(type), GetAdditionalWorldScope()); + else + return platform.GetReader(file.GetPath(type), "f"); } // static @@ -408,11 +403,9 @@ string CountryIndexes::IndexesDir(LocalCountryFile const & localFile) string dir = localFile.GetDirectory(); CountryFile const & file = localFile.GetCountryFile(); - /// @todo It's a temporary code until we will put fIndex->fOffset into mwm container. - /// IndexesDir should not throw any exceptions. - if (dir.empty()) + if (localFile.IsInBundle()) { - // Local file is stored in resources. Need to prepare index folder in the writable directory. + // Local file is stored in bundle. Need to prepare index folder in the writable directory. int64_t const version = localFile.GetVersion(); ASSERT_GREATER(version, 0, ()); diff --git a/storage/country_info_getter.cpp b/storage/country_info_getter.cpp index a72adaeff4..cf881069f0 100644 --- a/storage/country_info_getter.cpp +++ b/storage/country_info_getter.cpp @@ -148,9 +148,16 @@ m2::RectD CountryInfoGetter::CalcLimitRect(std::string const & prefix) const m2::RectD CountryInfoGetter::GetLimitRectForLeaf(CountryId const & leafCountryId) const { auto const it = m_countryIndex.find(leafCountryId); - ASSERT(it != m_countryIndex.end(), ()); - ASSERT_LESS(it->second, m_countries.size(), ()); - return m_countries[it->second].m_rect; + if (it != m_countryIndex.end()) + { + ASSERT_LESS(it->second, m_countries.size(), ()); + return m_countries[it->second].m_rect; + } + else + { + // Full rect for World files. + return mercator::Bounds::FullRect(); + } } void CountryInfoGetter::GetMatchedRegions(std::string const & affiliation, diff --git a/storage/storage.cpp b/storage/storage.cpp index dc084d7a29..322fcdbc79 100644 --- a/storage/storage.cpp +++ b/storage/storage.cpp @@ -14,6 +14,7 @@ #include "platform/preferred_languages.hpp" #include "platform/settings.hpp" +#include "coding/file_reader.hpp" #include "coding/file_writer.hpp" #include "coding/internal/file_data.hpp" #include "coding/sha1.hpp" @@ -202,7 +203,83 @@ void Storage::Clear() SaveDownloadQueue(); } -void Storage::RegisterAllLocalMaps(bool enableDiffs) +bool Storage::GetForceDownloadWorlds(std::vector & res) const +{ + CHECK_THREAD_CHECKER(m_threadChecker, ()); + + bool hasWorld[] = { false, false }; + string const worldName[] = { WORLD_FILE_NAME, WORLD_COASTS_FILE_NAME }; + + { + // Check if Worlds already present. + std::vector localFiles; + FindAllLocalMapsAndCleanup(m_currentVersion, m_dataDir, localFiles); + for (auto const & f : localFiles) + { + for (int i = 0; i < 2; ++i) + { + if (f.GetCountryName() == worldName[i]) + hasWorld[i] = true; + } + } + + if (hasWorld[0] && hasWorld[1]) + return false; + } + + Platform & pl = GetPlatform(); + + // Parse root data folder (we stored Worlds here in older versions). + // Note that m_dataDir maybe empty and we take Platform::WritableDir. + std::vector rootFiles; + (void)FindAllLocalMapsInDirectoryAndCleanup(m_dataDir.empty() ? pl.WritableDir() : m_dataDir, + 0 /* version */, -1 /* latestVersion */, rootFiles); + + bool anyWorldWasMoved = false; + for (auto const & f : rootFiles) + { + for (int i = 0; i < 2; ++i) + { + if (f.GetCountryName() != worldName[i]) + continue; + + if (!hasWorld[i]) + { + try + { + auto const filePath = f.GetPath(MapFileType::Map); + uint32_t const version = version::ReadVersionDate(pl.GetReader(filePath, "f")); + ASSERT(version > 0, ()); + + auto const dirPath = base::JoinPath(f.GetDirectory(), std::to_string(version)); + if (pl.MkDirChecked(dirPath) && + base::RenameFileX(filePath, base::JoinPath(dirPath, worldName[i] + DATA_FILE_EXTENSION))) + { + anyWorldWasMoved = hasWorld[i] = true; + break; + } + else + LOG(LERROR, ("Can't move", filePath, "into", dirPath)); + } + catch (RootException const & ex) + { + LOG(LERROR, ("Corrupted World file", ex.Msg())); + } + } + DeleteFromDiskWithIndexes(f, MapFileType::Map); + } + } + + for (int i = 0; i < 2; ++i) + { + if (!hasWorld[i]) + res.push_back(GetCountryFile(worldName[i])); + } + + return anyWorldWasMoved; +} + +void Storage::RegisterAllLocalMaps(bool enableDiffs /* = false */) { CHECK_THREAD_CHECKER(m_threadChecker, ()); @@ -210,25 +287,20 @@ void Storage::RegisterAllLocalMaps(bool enableDiffs) m_localFilesForFakeCountries.clear(); vector localFiles; - FindAllLocalMapsAndCleanup(GetCurrentDataVersion(), m_dataDir, localFiles); + FindAllLocalMapsAndCleanup(m_currentVersion, m_dataDir, localFiles); - auto compareByCountryAndVersion = [](LocalCountryFile const & lhs, LocalCountryFile const & rhs) { + sort(localFiles.begin(), localFiles.end(), [](LocalCountryFile const & lhs, LocalCountryFile const & rhs) + { if (lhs.GetCountryFile() != rhs.GetCountryFile()) return lhs.GetCountryFile() < rhs.GetCountryFile(); return lhs.GetVersion() > rhs.GetVersion(); - }; - - auto equalByCountry = [](LocalCountryFile const & lhs, LocalCountryFile const & rhs) { - return lhs.GetCountryFile() == rhs.GetCountryFile(); - }; - - sort(localFiles.begin(), localFiles.end(), compareByCountryAndVersion); + }); auto i = localFiles.begin(); while (i != localFiles.end()) { auto j = i + 1; - while (j != localFiles.end() && equalByCountry(*i, *j)) + while (j != localFiles.end() && i->GetCountryFile() == j->GetCountryFile()) { LocalCountryFile & localFile = *j; LOG(LINFO, ("Removing obsolete", localFile)); @@ -405,7 +477,7 @@ Status Storage::CountryStatusEx(CountryId const & countryId) const return status; auto localFile = GetLatestLocalFile(countryId); - if (!localFile || !localFile->OnDisk(MapFileType::Map)) + if (!localFile || !(localFile->OnDisk(MapFileType::Map) || localFile->IsInBundle())) return Status::NotDownloaded; auto const & countryFile = GetCountryFile(countryId); @@ -731,9 +803,16 @@ void Storage::RegisterDownloadedFiles(CountryId const & countryId, MapFileType t ASSERT_EQUAL(type, MapFileType::Map, ()); CountryFile const countryFile = GetCountryFile(countryId); - LocalFilePtr localFile = GetLocalFile(countryId, GetCurrentDataVersion()); - if (!localFile) - localFile = PreparePlaceForCountryFiles(GetCurrentDataVersion(), m_dataDir, countryFile); + LocalFilePtr localFile = GetLocalFile(countryId, m_currentVersion); + + if (!localFile || localFile->IsInBundle()) + localFile = PreparePlaceForCountryFiles(m_currentVersion, m_dataDir, countryFile); + else + { + /// @todo If localFile already exists, we will remove it from disk below? + LOG(LERROR, ("Downloaded country file for already existing one", *localFile)); + } + if (!localFile) { LOG(LERROR, ("Can't prepare LocalCountryFile for", countryFile, "in folder", m_dataDir)); @@ -798,13 +877,9 @@ void Storage::GetOutdatedCountries(vector & countries) const for (auto const & p : m_localFiles) { CountryId const & countryId = p.first; - string const name = GetCountryFile(countryId).GetName(); LocalFilePtr file = GetLatestLocalFile(countryId); - if (file && file->GetVersion() != GetCurrentDataVersion() && name != WORLD_COASTS_FILE_NAME - && name != WORLD_FILE_NAME) - { + if (file && file->GetVersion() != m_currentVersion && !IsWorldCountryID(countryId)) countries.push_back(&CountryLeafByCountryId(countryId)); - } } } */ @@ -876,7 +951,12 @@ void Storage::RegisterCountryFiles(LocalFilePtr localFile) { LocalFilePtr existingFile = GetLocalFile(countryId, localFile->GetVersion()); if (existingFile) - ASSERT_EQUAL(localFile.get(), existingFile.get(), ()); + { + if (existingFile->IsInBundle()) + *existingFile = *localFile; + else + ASSERT_EQUAL(localFile.get(), existingFile.get(), ()); + } else m_localFiles[countryId].push_front(localFile); } @@ -1133,12 +1213,22 @@ void Storage::GetChildrenInGroups(CountryId const & parent, CountriesVec & downl CountriesVec disputedTerritoriesWithoutSiblings; // All disputed territories in subtree with root == |parent|. CountriesVec allDisputedTerritories; - parentNode->ForEachChild([&](CountryTree::Node const & childNode) { + parentNode->ForEachChild([&](CountryTree::Node const & childNode) + { + CountryId const & childValue = childNode.Value().Name(); + + // Do not show bundled World files in Downloader UI, they are always exist and up-to-date. + if (IsWorldCountryID(childValue)) + { + auto const pFile = GetLatestLocalFile(childValue); + if (pFile && pFile->IsInBundle()) + return; + } + vector> disputedTerritoriesAndStatus; StatusAndError const childStatus = GetNodeStatusInfo(childNode, disputedTerritoriesAndStatus, true /* isDisputedTerritoriesCounted */); - CountryId const & childValue = childNode.Value().Name(); ASSERT_NOT_EQUAL(childStatus.status, NodeStatus::Undefined, ()); for (auto const & disputed : disputedTerritoriesAndStatus) allDisputedTerritories.push_back(disputed.first); @@ -1269,6 +1359,16 @@ bool Storage::IsDisputed(CountryTree::Node const & node) const return found.size() > 1; } +bool Storage::IsCountryLeaf(CountryTree::Node const & node) +{ + return (node.ChildrenCount() == 0 && !IsWorldCountryID(node.Value().Name())); +} + +bool Storage::IsWorldCountryID(CountryId const & country) +{ + return strings::StartsWith(country, WORLD_FILE_NAME); +} + void Storage::CalcMaxMwmSizeBytes() { m_maxMwmSizeBytes = 0; diff --git a/storage/storage.hpp b/storage/storage.hpp index 830c0233aa..3432f829cf 100644 --- a/storage/storage.hpp +++ b/storage/storage.hpp @@ -479,6 +479,10 @@ public: // Clears local files registry and downloader's queue. void Clear(); + /// Used in Android to check Worlds to download. + /// @return true If any World file was moved. + bool GetForceDownloadWorlds(std::vector & res) const; + // Finds and registers all map files in maps directory. In the case // of several versions of the same map keeps only the latest one, others // are deleted from disk. @@ -634,6 +638,10 @@ private: /// @return true if |node.Value().Name()| is a disputed territory and false otherwise. bool IsDisputed(CountryTree::Node const & node) const; + /// @return true iff \a node is a country MWM leaf of the tree. + static bool IsCountryLeaf(CountryTree::Node const & node); + static bool IsWorldCountryID(CountryId const & country); + void CalcMaxMwmSizeBytes(); void OnMapDownloadFailed(CountryId const & countryId); @@ -713,7 +721,7 @@ void Storage::ForEachCountry(ToDo && toDo) const { m_countries.GetRoot().ForEachInSubtree([&](CountryTree::Node const & node) { - if (node.ChildrenCount() == 0) + if (IsCountryLeaf(node)) toDo(node.Value()); }); } diff --git a/storage/storage_tests/CMakeLists.txt b/storage/storage_tests/CMakeLists.txt index 2cebfe6c5c..3233d526f3 100644 --- a/storage/storage_tests/CMakeLists.txt +++ b/storage/storage_tests/CMakeLists.txt @@ -1,6 +1,7 @@ project(storage_tests) set(SRC + countries_tests.cpp country_info_getter_tests.cpp country_name_getter_tests.cpp downloader_tests.cpp diff --git a/storage/storage_tests/countries_tests.cpp b/storage/storage_tests/countries_tests.cpp new file mode 100644 index 0000000000..4478a6ca97 --- /dev/null +++ b/storage/storage_tests/countries_tests.cpp @@ -0,0 +1,14 @@ +#include "testing/testing.hpp" + +#include "platform/platform.hpp" + +#include "coding/sha1.hpp" + +#include "base/logging.hpp" + +UNIT_TEST(CalculateWorldSHA) +{ + auto const path = GetPlatform().ResourcesDir(); + for (char const * country : { WORLD_FILE_NAME, WORLD_COASTS_FILE_NAME }) + LOG(LINFO, (country, coding::SHA1::CalculateBase64(path + country + DATA_FILE_EXTENSION))); +} diff --git a/storage/storage_tests/storage_tests.cpp b/storage/storage_tests/storage_tests.cpp index 4566b1c1c3..544ce31939 100644 --- a/storage/storage_tests/storage_tests.cpp +++ b/storage/storage_tests/storage_tests.cpp @@ -797,29 +797,26 @@ UNIT_CLASS_TEST(StorageTest, DownloadedMap) } // Storage::GetLocalRealMaps() test. - /* - CountriesVec localRealMaps; - storage.GetLocalRealMaps(localRealMaps); - TEST_EQUAL(localRealMaps.size(), 4, ()); +// CountriesVec localRealMaps; +// storage.GetLocalRealMaps(localRealMaps); +// TEST_EQUAL(localRealMaps.size(), 4, ()); TEST(storage.IsNodeDownloaded("Algeria_Central"), ()); TEST(storage.IsNodeDownloaded("Algeria_Coast"), ()); TEST(!storage.IsNodeDownloaded("Algeria_Coast.mwm"), ()); - TEST(!storage.IsNodeDownloaded(WORLD_FILE_NAME), ()); - TEST(!storage.IsNodeDownloaded(WORLD_COASTS_FILE_NAME), ()); - */ + TEST(storage.IsNodeDownloaded(WORLD_FILE_NAME), ()); + TEST(storage.IsNodeDownloaded(WORLD_COASTS_FILE_NAME), ()); // Storage::GetChildrenInGroups test when at least Algeria_Central and Algeria_Coast have been downloaded. CountryId const rootCountryId = storage.GetRootId(); TEST_EQUAL(rootCountryId, COUNTRIES_ROOT, ()); CountriesVec downloaded, available; - CountriesVec downloadedWithKeep, availableWithKeep; - storage.GetChildrenInGroups(rootCountryId, downloaded, available); TEST_EQUAL(downloaded.size(), 1, (downloaded)); TEST_EQUAL(available.size(), 223, ()); + CountriesVec downloadedWithKeep, availableWithKeep; storage.GetChildrenInGroups(rootCountryId, downloadedWithKeep, availableWithKeep, true /* keepAvailableChildren*/); TEST_EQUAL(downloadedWithKeep.size(), 1, (downloadedWithKeep));