diff --git a/platform/downloader_defines.hpp b/platform/downloader_defines.hpp index 4ca46c94d1..399dd4a149 100644 --- a/platform/downloader_defines.hpp +++ b/platform/downloader_defines.hpp @@ -13,7 +13,8 @@ enum class DownloadStatus InProgress, Completed, Failed, - FileNotFound + FileNotFound, + FailedSHA, }; inline std::string DebugPrint(DownloadStatus status) @@ -24,6 +25,7 @@ inline std::string DebugPrint(DownloadStatus status) case DownloadStatus::Completed: return "Completed"; case DownloadStatus::Failed: return "Failed"; case DownloadStatus::FileNotFound: return "File not found"; + case DownloadStatus::FailedSHA: return "Failed SHA check"; } UNREACHABLE(); } diff --git a/storage/storage.cpp b/storage/storage.cpp index 52f3a48aab..dc4e64307a 100644 --- a/storage/storage.cpp +++ b/storage/storage.cpp @@ -16,6 +16,7 @@ #include "coding/file_writer.hpp" #include "coding/internal/file_data.hpp" +#include "coding/sha1.hpp" #include "base/exception.hpp" #include "base/file_name_utils.hpp" @@ -659,9 +660,45 @@ void Storage::OnDownloadFinished(QueuedCountry const & queuedCountry, DownloadSt m_downloadingCountries.erase(queuedCountry.GetCountryId()); - OnMapDownloadFinished(queuedCountry.GetCountryId(), status, queuedCountry.GetFileType()); + auto const & countryId = queuedCountry.GetCountryId(); + auto const fileType = queuedCountry.GetFileType(); + auto const finishFn = [this, countryId, fileType] (DownloadStatus status) + { + OnMapDownloadFinished(countryId, status, fileType); + OnFinishDownloading(); + }; - OnFinishDownloading(); + if (status == DownloadStatus::Completed && m_integrityValidationEnabled) + { + /// @todo Can/Should be combined with ApplyDiff routine when we will restore it. + /// While this is simple and working solution, I think that Downloader component + /// should make this kind of checks (taking expecting SHA as input). But now it's + /// not so simple as it may seem .. + + GetPlatform().RunTask(Platform::Thread::File, [path = GetFileDownloadPath(countryId, fileType), + sha1 = GetCountryFile(countryId).GetSha1(), + fn = std::move(finishFn)]() + { + DownloadStatus status = DownloadStatus::Completed; + + if (coding::SHA1::CalculateBase64(path) != sha1) + { + LOG(LERROR, ("SHA check error for", path)); + base::DeleteFileX(path); + status = DownloadStatus::FailedSHA; + } + + GetPlatform().RunTask(Platform::Thread::Gui, [fn = std::move(fn), status]() + { + if (status == DownloadStatus::Completed) + LOG(LDEBUG, ("Successful SHA check")); + + fn(status); + }); + }); + } + else + finishFn(status); } void Storage::RegisterDownloadedFiles(CountryId const & countryId, MapFileType type) @@ -721,13 +758,6 @@ void Storage::RegisterDownloadedFiles(CountryId const & countryId, MapFileType t return; } - if (m_integrityValidationEnabled && !localFile->ValidateIntegrity()) - { - base::DeleteFileX(localFile->GetPath(MapFileType::Map)); - fn(false /* isSuccess */); - return; - } - RegisterCountryFiles(localFile); fn(true); }