diff --git a/storage/http_map_files_downloader.cpp b/storage/http_map_files_downloader.cpp new file mode 100644 index 0000000000..2a55c9a1db --- /dev/null +++ b/storage/http_map_files_downloader.cpp @@ -0,0 +1,78 @@ +#include "storage/http_map_files_downloader.hpp" + +#include "platform/platform.hpp" +#include "platform/servers_list.hpp" + +#include "base/assert.hpp" + +#include "std/bind.hpp" + +namespace storage +{ +HttpMapFilesDownloader::~HttpMapFilesDownloader() +{ + ASSERT(m_checker.CalledOnOriginalThread(), ()); +} + +void HttpMapFilesDownloader::GetServersList(string const & mapFileName, + TServersListCallback const & callback) +{ + ASSERT(m_checker.CalledOnOriginalThread(), ()); + m_request.reset(downloader::HttpRequest::PostJson( + GetPlatform().MetaServerUrl(), mapFileName, + bind(&HttpMapFilesDownloader::OnServersListDownloaded, this, callback, _1))); +} + +void HttpMapFilesDownloader::DownloadMapFile(vector const & urls, string const & path, + int64_t size, + TFileDownloadedCallback const & onDownloaded, + TDownloadingProgressCallback const & onProgress) +{ + ASSERT(m_checker.CalledOnOriginalThread(), ()); + m_request.reset(downloader::HttpRequest::GetFile( + urls, path, size, bind(&HttpMapFilesDownloader::OnMapFileDownloaded, this, onDownloaded, _1), + bind(&HttpMapFilesDownloader::OnMapFileDownloadingProgress, this, onProgress, _1))); +} + +MapFilesDownloader::TProgress HttpMapFilesDownloader::GetDownloadingProgress() +{ + ASSERT(m_checker.CalledOnOriginalThread(), ()); + return m_request->Progress(); +} + +bool HttpMapFilesDownloader::IsIdle() +{ + ASSERT(m_checker.CalledOnOriginalThread(), ()); + return m_request.get() != nullptr; +} + +void HttpMapFilesDownloader::Reset() +{ + ASSERT(m_checker.CalledOnOriginalThread(), ()); + m_request.reset(); +} + +void HttpMapFilesDownloader::OnServersListDownloaded(TServersListCallback const & callback, + downloader::HttpRequest & request) +{ + ASSERT(m_checker.CalledOnOriginalThread(), ()); + vector urls; + GetServerListFromRequest(request, urls); + callback(urls); +} + +void HttpMapFilesDownloader::OnMapFileDownloaded(TFileDownloadedCallback const & onDownloaded, + downloader::HttpRequest & request) +{ + ASSERT(m_checker.CalledOnOriginalThread(), ()); + bool const success = request.Status() != downloader::HttpRequest::EFailed; + onDownloaded(success, request.Progress()); +} + +void HttpMapFilesDownloader::OnMapFileDownloadingProgress( + TDownloadingProgressCallback const & onProgress, downloader::HttpRequest & request) +{ + ASSERT(m_checker.CalledOnOriginalThread(), ()); + onProgress(request.Progress()); +} +} // namespace storage diff --git a/storage/http_map_files_downloader.hpp b/storage/http_map_files_downloader.hpp new file mode 100644 index 0000000000..21d5004919 --- /dev/null +++ b/storage/http_map_files_downloader.hpp @@ -0,0 +1,40 @@ +#pragma once + +#include "storage/map_files_downloader.hpp" +#include "platform/http_request.hpp" +#include "base/thread_checker.hpp" +#include "std/unique_ptr.hpp" + +namespace storage +{ +/// This class encapsulates HTTP requests for receiving server lists +/// and file downloading. +// +// *NOTE*, this class is not thread-safe. +class HttpMapFilesDownloader : public MapFilesDownloader +{ +public: + virtual ~HttpMapFilesDownloader(); + + // MapFilesDownloader overrides: + void GetServersList(string const & mapFileName, TServersListCallback const & callback) override; + void DownloadMapFile(vector const & urls, string const & path, int64_t size, + TFileDownloadedCallback const & onDownloaded, + TDownloadingProgressCallback const & onProgress) override; + TProgress GetDownloadingProgress() override; + bool IsIdle() override; + void Reset() override; + +private: + void OnServersListDownloaded(TServersListCallback const & callback, + downloader::HttpRequest & request); + void OnMapFileDownloaded(TFileDownloadedCallback const & onDownloaded, + downloader::HttpRequest & request); + void OnMapFileDownloadingProgress(TDownloadingProgressCallback const & onProgress, + downloader::HttpRequest & request); + + unique_ptr m_request; + + ThreadChecker m_checker; +}; +} // namespace storage diff --git a/storage/map_files_downloader.hpp b/storage/map_files_downloader.hpp new file mode 100644 index 0000000000..5c888b1d78 --- /dev/null +++ b/storage/map_files_downloader.hpp @@ -0,0 +1,44 @@ +#pragma once + +#include "std/function.hpp" +#include "std/string.hpp" +#include "std/vector.hpp" +#include "std/utility.hpp" + +namespace storage +{ +// This interface encapsulates HTTP routines for receiving servers +// URLs and downloading a single map file. +class MapFilesDownloader +{ +public: + using TProgress = pair; + + using TFileDownloadedCallback = function; + using TDownloadingProgressCallback = function; + using TServersListCallback = function & urls)>; + + virtual ~MapFilesDownloader() = default; + + /// Asynchroniously receives a list of all servers that can be asked + /// for a map file and invokes callback on the original thread. + virtual void GetServersList(string const & mapFileName, + TServersListCallback const & callback) = 0; + + /// Asynchroniously downloads a map file, periodically invokes + /// onProgress callback and finally invokes onDownloaded + /// callback. Both callbacks will be invoked on the original thread. + virtual void DownloadMapFile(vector const & urls, string const & path, int64_t size, + TFileDownloadedCallback const & onDownloaded, + TDownloadingProgressCallback const & onProgress) = 0; + + /// Returns current downloading progress. + virtual TProgress GetDownloadingProgress() = 0; + + /// Returns true when downloader does not perform any job. + virtual bool IsIdle() = 0; + + /// Resets downloader to the idle state. + virtual void Reset() = 0; +}; +} // namespace storage diff --git a/storage/storage.cpp b/storage/storage.cpp index f0e4de8abd..b6557bad02 100644 --- a/storage/storage.cpp +++ b/storage/storage.cpp @@ -1,5 +1,7 @@ #include "storage/storage.hpp" +#include "storage/http_map_files_downloader.hpp" + #include "defines.hpp" #include "platform/platform.hpp" @@ -173,7 +175,7 @@ namespace storage } - Storage::Storage() : m_currentSlotId(0) + Storage::Storage() : m_downloader(new HttpMapFilesDownloader()), m_currentSlotId(0) { LoadCountriesFile(false); @@ -266,8 +268,8 @@ namespace storage if (found != m_queue.end()) { sizes.second = cnt.GetFullRemoteSize(); - if (m_request != nullptr && m_queue.front().GetIndex() == index) - sizes.first = m_request->Progress().first + m_countryProgress.first; + if (!m_downloader->IsIdle() && m_queue.front().GetIndex() == index) + sizes.first = m_downloader->GetDownloadingProgress().first + m_countryProgress.first; } else sizes = cnt.GetFullSize(); @@ -371,9 +373,8 @@ namespace storage void Storage::DownloadNextFile(QueuedCountry const & cnt) { // send Country name for statistics - m_request.reset(HttpRequest::PostJson(GetPlatform().MetaServerUrl(), - cnt.GetFileName(), - bind(&Storage::OnServerListDownloaded, this, _1))); + m_downloader->GetServersList(cnt.GetFileName(), + bind(&Storage::OnServerListDownloaded, this, _1)); } /* @@ -393,7 +394,7 @@ namespace storage if (found == m_queue.begin()) { // stop download - m_request.reset(); + m_downloader->Reset(); // remove from the queue m_queue.erase(found); // start another download if the queue is not empty @@ -458,7 +459,7 @@ namespace storage } } - void Storage::OnMapDownloadFinished(HttpRequest & request) + void Storage::OnMapDownloadFinished(bool success, MapFilesDownloader::TProgress const & progress) { if (m_queue.empty()) { @@ -469,7 +470,7 @@ namespace storage QueuedCountry & cnt = m_queue.front(); TIndex const index = cnt.GetIndex(); - bool const downloadHasFailed = (request.Status() == HttpRequest::EFailed); + bool const downloadHasFailed = !success; { string optionName; switch (cnt.GetInitOptions()) @@ -492,11 +493,10 @@ namespace storage } else { - HttpRequest::ProgressT const & p = request.Progress(); - ASSERT_EQUAL(p.first, p.second, ()); - ASSERT_EQUAL(p.first, cnt.GetDownloadSize(), ()); + ASSERT_EQUAL(progress.first, progress.second, ()); + ASSERT_EQUAL(progress.first, cnt.GetDownloadSize(), ()); - m_countryProgress.first += p.first; + m_countryProgress.first += progress.first; if (cnt.MoveNextFile()) { DownloadNextFile(cnt); @@ -511,7 +511,7 @@ namespace storage NotifyStatusChanged(index); - m_request.reset(); + m_downloader->Reset(); DownloadNextCountryFromQueue(); } @@ -521,7 +521,7 @@ namespace storage o.m_progressFn(idx, p); } - void Storage::OnMapDownloadProgress(HttpRequest & request) + void Storage::OnMapDownloadProgress(MapFilesDownloader::TProgress const & progress) { if (m_queue.empty()) { @@ -531,7 +531,7 @@ namespace storage if (!m_observers.empty()) { - HttpRequest::ProgressT p = request.Progress(); + MapFilesDownloader::TProgress p = progress; p.first += m_countryProgress.first; p.second = m_countryProgress.second; @@ -539,7 +539,7 @@ namespace storage } } - void Storage::OnServerListDownloaded(HttpRequest & request) + void Storage::OnServerListDownloaded(vector const & urls) { if (m_queue.empty()) { @@ -549,19 +549,16 @@ namespace storage QueuedCountry const & cnt = m_queue.front(); - vector urls; - GetServerListFromRequest(request, urls); - + vector fileUrls(urls.size()); // append actual version and file name string const fileName = cnt.GetFileName(); for (size_t i = 0; i < urls.size(); ++i) - urls[i] = GetFileDownloadUrl(urls[i], fileName); + fileUrls[i] = GetFileDownloadUrl(urls[i], fileName); string const filePath = GetPlatform().WritablePathForFile(fileName + READY_FILE_EXTENSION); - m_request.reset(HttpRequest::GetFile(urls, filePath, cnt.GetDownloadSize(), - bind(&Storage::OnMapDownloadFinished, this, _1), - bind(&Storage::OnMapDownloadProgress, this, _1))); - ASSERT ( m_request, () ); + m_downloader->DownloadMapFile(fileUrls, filePath, cnt.GetDownloadSize(), + bind(&Storage::OnMapDownloadFinished, this, _1, _2), + bind(&Storage::OnMapDownloadProgress, this, _1)); } string Storage::GetFileDownloadUrl(string const & baseUrl, string const & fName) const @@ -660,6 +657,11 @@ namespace storage res.push_back(&CountryByIndex(FindIndexByFile(fList[i]))); } + void Storage::SetDownloaderForTesting(unique_ptr && downloader) + { + m_downloader = move(downloader); + } + TStatus Storage::CountryStatusWithoutFailed(TIndex const & index) const { // first, check if we already downloading this country or have in in the queue diff --git a/storage/storage.hpp b/storage/storage.hpp index 60c1f5461d..2ccdd53d6d 100644 --- a/storage/storage.hpp +++ b/storage/storage.hpp @@ -1,10 +1,9 @@ #pragma once -#include "storage/storage_defines.hpp" #include "storage/country.hpp" #include "storage/index.hpp" - -#include "platform/http_request.hpp" +#include "storage/map_files_downloader.hpp" +#include "storage/storage_defines.hpp" #include "std/vector.hpp" #include "std/list.hpp" @@ -20,7 +19,7 @@ namespace storage class Storage { /// We support only one simultaneous request at the moment - unique_ptr m_request; + unique_ptr m_downloader; /// stores timestamp for update checks int64_t m_currentVersion; @@ -67,7 +66,7 @@ namespace storage /// used to correctly calculate total country download progress with more than 1 file /// - downloader::HttpRequest::ProgressT m_countryProgress; + MapFilesDownloader::TProgress m_countryProgress; /// @name Communicate with GUI //@{ @@ -101,9 +100,9 @@ namespace storage /// @name //@{ - void OnServerListDownloaded(downloader::HttpRequest & request); - void OnMapDownloadFinished(downloader::HttpRequest & request); - void OnMapDownloadProgress(downloader::HttpRequest & request); + void OnServerListDownloaded(vector const & urls); + void OnMapDownloadFinished(bool success, MapFilesDownloader::TProgress const & progress); + void OnMapDownloadProgress(MapFilesDownloader::TProgress const & progress); void DownloadNextFile(QueuedCountry const & cnt); //@} @@ -155,6 +154,8 @@ namespace storage int64_t GetCurrentDataVersion() const { return m_currentVersion; } + void SetDownloaderForTesting(unique_ptr && downloader); + private: TStatus CountryStatusWithoutFailed(TIndex const & index) const; TStatus CountryStatusFull(TIndex const & index, TStatus const status) const; diff --git a/storage/storage.pro b/storage/storage.pro index 01f5e4bd07..ba4d116a3e 100644 --- a/storage/storage.pro +++ b/storage/storage.pro @@ -12,17 +12,21 @@ INCLUDEPATH += $$ROOT_DIR/3party/jansson/src HEADERS += \ country.hpp \ + country_decl.hpp \ + country_info.hpp \ + country_polygon.hpp \ + http_map_files_downloader.hpp \ + index.hpp \ + map_files_downloader.hpp \ simple_tree.hpp \ storage.hpp \ - country_polygon.hpp \ - country_info.hpp \ - country_decl.hpp \ - index.hpp \ storage_defines.hpp \ SOURCES += \ country.cpp \ - storage.cpp \ - country_info.cpp \ country_decl.cpp \ + country_info.cpp \ + http_map_files_downloader.cpp \ index.cpp \ + storage.cpp \ + storage_defines.cpp \ diff --git a/storage/storage_defines.cpp b/storage/storage_defines.cpp new file mode 100644 index 0000000000..cb50d6c432 --- /dev/null +++ b/storage/storage_defines.cpp @@ -0,0 +1,27 @@ +#include "storage/storage_defines.hpp" + +namespace storage +{ +string DebugPrint(TStatus status) +{ + switch (status) + { + case TStatus::EOnDisk: + return "OnDisk"; + case TStatus::ENotDownloaded: + return "NotDownloaded"; + case TStatus::EDownloadFailed: + return "DownloadFailed"; + case TStatus::EDownloading: + return "Downloading"; + case TStatus::EInQueue: + return "InQueue"; + case TStatus::EUnknown: + return "Unknown"; + case TStatus::EOnDiskOutOfDate: + return "OnDiskOutOfDate"; + case TStatus::EOutOfMemFailed: + return "OutOfMemFailed"; + } +} +} // namespace storage diff --git a/storage/storage_defines.hpp b/storage/storage_defines.hpp index ca89790d28..9cea845c27 100644 --- a/storage/storage_defines.hpp +++ b/storage/storage_defines.hpp @@ -3,6 +3,7 @@ #include "3party/enum_flags.hpp" #include "std/stdint.hpp" +#include "std/string.hpp" #include "std/utility.hpp" namespace storage @@ -20,6 +21,8 @@ namespace storage EOutOfMemFailed // EDownloadFailed because not enough memory }; + string DebugPrint(TStatus status); + ENUM_FLAGS(TMapOptions) enum class TMapOptions { diff --git a/storage/storage_tests/fake_map_files_downloader.cpp b/storage/storage_tests/fake_map_files_downloader.cpp new file mode 100644 index 0000000000..bfd116f6a4 --- /dev/null +++ b/storage/storage_tests/fake_map_files_downloader.cpp @@ -0,0 +1,78 @@ +#include "storage/storage_tests/fake_map_files_downloader.hpp" + +#include "storage/storage_tests/message_loop.hpp" + +#include "coding/file_writer.hpp" +#include "base/assert.hpp" +#include "base/scope_guard.hpp" +#include "std/algorithm.hpp" + +namespace storage +{ +namespace +{ +int64_t const kBlockSize = 1024 * 1024; +} // namespace + +FakeMapFilesDownloader::FakeMapFilesDownloader(MessageLoop & loop) + : m_progress(make_pair(0, 0)), m_idle(true), m_loop(loop) +{ + m_servers.push_back("http://test-url/"); +} + +FakeMapFilesDownloader::~FakeMapFilesDownloader() +{ + CHECK(m_checker.CalledOnOriginalThread(), ()); +} + +void FakeMapFilesDownloader::GetServersList(string const & mapFileName, + TServersListCallback const & callback) +{ + CHECK(m_checker.CalledOnOriginalThread(), ()); + m_idle = false; + MY_SCOPE_GUARD(resetIdle, [this]() { m_idle = true; }); + m_loop.PostTask(bind(callback, m_servers)); +} + +void FakeMapFilesDownloader::DownloadMapFile(vector const & urls, string const & path, + int64_t size, + TFileDownloadedCallback const & onDownloaded, + TDownloadingProgressCallback const & onProgress) +{ + static string kZeroes(kBlockSize, '\0'); + + m_progress.first = 0; + m_progress.second = size; + m_idle = false; + MY_SCOPE_GUARD(resetIdle, [this]() { m_idle = true; }); + + { + FileWriter writer(path); + while (size != 0) + { + int64_t const blockSize = min(size, kBlockSize); + writer.Write(kZeroes.data(), blockSize); + size -= blockSize; + m_progress.first += blockSize; + m_loop.PostTask(bind(onProgress, m_progress)); + } + } + m_loop.PostTask(bind(onDownloaded, true /* success */, m_progress)); +} + +MapFilesDownloader::TProgress FakeMapFilesDownloader::GetDownloadingProgress() +{ + return m_progress; +} + +bool FakeMapFilesDownloader::IsIdle() { + CHECK(m_checker.CalledOnOriginalThread(), ()); + return m_idle; +} + +void FakeMapFilesDownloader::Reset() { + CHECK(m_checker.CalledOnOriginalThread(), ()); + m_idle = true; +} + +} // namespace storage diff --git a/storage/storage_tests/fake_map_files_downloader.hpp b/storage/storage_tests/fake_map_files_downloader.hpp new file mode 100644 index 0000000000..fb9ec9f19f --- /dev/null +++ b/storage/storage_tests/fake_map_files_downloader.hpp @@ -0,0 +1,40 @@ +#pragma once + +#include "storage/map_files_downloader.hpp" +#include "base/thread_checker.hpp" + +namespace storage +{ +class MessageLoop; + +// This class can be used in tests to mimic a real downloader. It +// always returns a single URL for map files downloading and when +// asked for a file, creates a file with zero-bytes content on a disk. +// Because all callbacks must be invoked asynchroniously, it needs a +// single-thread message loop runner to run callbacks. +// +// *NOTE*, this class is not thread-safe. +class FakeMapFilesDownloader : public MapFilesDownloader +{ +public: + FakeMapFilesDownloader(MessageLoop & loop); + virtual ~FakeMapFilesDownloader(); + + // MapFilesDownloader overrides: + void GetServersList(string const & mapFileName, TServersListCallback const & callback) override; + void DownloadMapFile(vector const & urls, string const & path, int64_t size, + TFileDownloadedCallback const & onDownloaded, + TDownloadingProgressCallback const & onProgress) override; + TProgress GetDownloadingProgress() override; + bool IsIdle() override; + void Reset() override; + + private: + vector m_servers; + TProgress m_progress; + bool m_idle; + + MessageLoop & m_loop; + ThreadChecker m_checker; +}; +} // namespace storage diff --git a/storage/storage_tests/message_loop.cpp b/storage/storage_tests/message_loop.cpp new file mode 100644 index 0000000000..da91b22341 --- /dev/null +++ b/storage/storage_tests/message_loop.cpp @@ -0,0 +1,28 @@ +#include "storage/storage_tests/message_loop.hpp" + +#include "base/assert.hpp" + +namespace storage +{ +MessageLoop::~MessageLoop() +{ + CHECK(m_checker.CalledOnOriginalThread(), ()); +} + +void MessageLoop::Run() +{ + CHECK(m_checker.CalledOnOriginalThread(), ()); + while (!m_tasks.empty()) + { + TTask task = m_tasks.front(); + m_tasks.pop(); + task(); + } +} + +void MessageLoop::PostTask(TTask const & task) +{ + CHECK(m_checker.CalledOnOriginalThread(), ()); + m_tasks.push(task); +} +} // namespace storage diff --git a/storage/storage_tests/message_loop.hpp b/storage/storage_tests/message_loop.hpp new file mode 100644 index 0000000000..b8e9eab1b4 --- /dev/null +++ b/storage/storage_tests/message_loop.hpp @@ -0,0 +1,32 @@ +#pragma once + +#include "base/thread_checker.hpp" + +#include "std/function.hpp" +#include "std/queue.hpp" + +namespace storage +{ +// This class can be used in tests to mimic asynchronious calls. For +// example, when task A invokes task B and passes a callback C as an +// argument to B, it's silly for B to call C directly if B is an +// asynchronious task. So, the solution is to post C on the same +// message loop where B is run. +// +// *NOTE*, this class is not thread-safe. +class MessageLoop +{ + public: + using TTask = function; + + ~MessageLoop(); + + void Run(); + void PostTask(TTask const & task); + + private: + queue m_tasks; + + ThreadChecker m_checker; +}; +} // namespace diff --git a/storage/storage_tests/storage_tests.cpp b/storage/storage_tests/storage_tests.cpp index 346b8a51e5..e66be7889a 100644 --- a/storage/storage_tests/storage_tests.cpp +++ b/storage/storage_tests/storage_tests.cpp @@ -1,12 +1,114 @@ #include "testing/testing.hpp" #include "storage/storage.hpp" +#include "storage/storage_defines.hpp" +#include "storage/storage_tests/fake_map_files_downloader.hpp" +#include "storage/storage_tests/message_loop.hpp" + +#include "platform/platform.hpp" + +#include "coding/file_name_utils.hpp" +#include "coding/file_writer.hpp" +#include "coding/internal/file_data.hpp" #include "defines.hpp" +#include "base/scope_guard.hpp" + +#include "std/bind.hpp" +#include "std/unique_ptr.hpp" using namespace storage; +namespace +{ +// This class checks steps Storage::DownloadMap() performs to download a map. +class CountryDownloaderChecker +{ +public: + CountryDownloaderChecker(Storage & storage, string const & countryFileName) + : m_storage(storage), + m_index(m_storage.FindIndexByFile(countryFileName)), + m_lastStatus(TStatus::ENotDownloaded), + m_bytesDownloaded(0), + m_totalBytesToDownload(-1) + { + CHECK(m_index.IsValid(), ()); + CHECK_EQUAL(m_lastStatus, m_storage.CountryStatusEx(m_index), ()); + m_slot = m_storage.Subscribe( + bind(&CountryDownloaderChecker::OnCountryStatusChanged, this, _1), + bind(&CountryDownloaderChecker::OnCountryDownloadingProgress, this, _1, _2)); + + m_storage.DownloadCountry(m_index, TMapOptions::EMap); + } + + ~CountryDownloaderChecker() + { + CHECK_EQUAL(TStatus::EOnDisk, m_lastStatus, ()); + CHECK_EQUAL(m_bytesDownloaded, m_totalBytesToDownload, ()); + m_storage.Unsubscribe(m_slot); + } + +private: + void OnCountryStatusChanged(TIndex const & index) + { + if (index != m_index) + return; + TStatus status = m_storage.CountryStatusEx(m_index); + + LOG(LINFO, ("OnCountryStatusChanged", status)); + switch (status) + { + case TStatus::EDownloading: + CHECK(m_lastStatus == TStatus::EDownloading || m_lastStatus == TStatus::ENotDownloaded, + ("It's only possible to move from {Downloading|NotDownloaded} to Downloading," + "old status:", + m_lastStatus, ", new status:", status)); + break; + case TStatus::EOnDisk: + CHECK(m_lastStatus == TStatus::EDownloading || m_lastStatus == TStatus::ENotDownloaded, + ("It's only possible to move from {Downloading|NotDownloaded} to OnDisk,", + "old status:", m_lastStatus, ", new status:", status)); + CHECK_EQUAL(m_totalBytesToDownload, m_bytesDownloaded, ()); + break; + default: + CHECK(false, ("Unknown state change: from", m_lastStatus, "to", status)); + } + + m_lastStatus = status; + } + + void OnCountryDownloadingProgress(TIndex const & index, LocalAndRemoteSizeT const & progress) + { + if (index != m_index) + return; + LOG(LINFO, ("OnCountryDownloadingProgress:", progress)); + CHECK_GREATER(progress.first, m_bytesDownloaded, ()); + m_bytesDownloaded = progress.first; + m_totalBytesToDownload = progress.second; + CHECK_LESS_OR_EQUAL(m_bytesDownloaded, m_totalBytesToDownload, ()); + } + + Storage & m_storage; + TIndex const m_index; + TStatus m_lastStatus; + int m_slot; + + int64_t m_bytesDownloaded; + int64_t m_totalBytesToDownload; +}; + +void OnCountryDownloaded(string const & mapFileName, TMapOptions files) +{ + LOG(LINFO, ("OnCountryDownloaded:", mapFileName, static_cast(files))); + Platform & platform = GetPlatform(); + CHECK(my::RenameFileX( + my::JoinFoldersToPath(platform.WritableDir(), mapFileName + READY_FILE_EXTENSION), + my::JoinFoldersToPath(platform.WritableDir(), mapFileName)), + ()); +} + +} // namespace UNIT_TEST(StorageTest_Smoke) { @@ -18,7 +120,37 @@ UNIT_TEST(StorageTest_Smoke) TIndex const i2 = st.FindIndexByFile("Georgia"); TEST(i2.IsValid(), ()); - TEST_EQUAL(st.CountryFileName(i2, TMapOptions::ECarRouting), "Georgia" DATA_FILE_EXTENSION ROUTING_FILE_EXTENSION, ()); + TEST_EQUAL(st.CountryFileName(i2, TMapOptions::ECarRouting), + "Georgia" DATA_FILE_EXTENSION ROUTING_FILE_EXTENSION, ()); TEST_NOT_EQUAL(i1, i2, ()); } + +UNIT_TEST(StorageTest_Download) +{ + Platform & platform = GetPlatform(); + + string const countryFileName = "Azerbaijan"; + string const localFileName = + my::JoinFoldersToPath(platform.WritableDir(), countryFileName + DATA_FILE_EXTENSION); + string const downloadedFileName = my::JoinFoldersToPath( + platform.WritableDir(), countryFileName + DATA_FILE_EXTENSION + READY_FILE_EXTENSION); + + TEST(!Platform::IsFileExistsByFullPath(localFileName), + ("Please, remove", localFileName, "before running the test.")); + TEST(!Platform::IsFileExistsByFullPath(downloadedFileName), + ("Please, remove", downloadedFileName, "before running the test.")); + MY_SCOPE_GUARD(removeLocalFile, bind(&FileWriter::DeleteFileX, localFileName)); + MY_SCOPE_GUARD(removeDownloadedFile, bind(&FileWriter::DeleteFileX, downloadedFileName)); + + Storage storage; + storage.Init(&OnCountryDownloaded); + + MessageLoop loop; + storage.SetDownloaderForTesting(make_unique(loop)); + + { + CountryDownloaderChecker checker(storage, countryFileName); + loop.Run(); + } +} diff --git a/storage/storage_tests/storage_tests.pro b/storage/storage_tests/storage_tests.pro index 3739473f21..c0394a310b 100644 --- a/storage/storage_tests/storage_tests.pro +++ b/storage/storage_tests/storage_tests.pro @@ -17,10 +17,15 @@ linux*|win32-msvc*: QT *= network QT *= core -HEADERS += +HEADERS += \ + fake_map_files_downloader.hpp \ + message_loop.hpp \ + SOURCES += \ ../../testing/testingmain.cpp \ - simple_tree_test.cpp \ country_info_test.cpp \ + fake_map_files_downloader.cpp \ + message_loop.cpp \ + simple_tree_test.cpp \ storage_tests.cpp \