diff --git a/coding/streams_sink.hpp b/coding/streams_sink.hpp index b27197a992..3e44c870c4 100644 --- a/coding/streams_sink.hpp +++ b/coding/streams_sink.hpp @@ -29,6 +29,11 @@ namespace stream t = ReadPrimitiveFromSource(m_reader); return (*this); } + SinkReaderStream & operator >> (uint16_t & t) + { + t = ReadPrimitiveFromSource(m_reader); + return (*this); + } SinkReaderStream & operator >> (int64_t & t) { t = ReadPrimitiveFromSource(m_reader); @@ -39,6 +44,11 @@ namespace stream t = ReadPrimitiveFromSource(m_reader); return (*this); } + SinkReaderStream & operator >> (int16_t & t) + { + t = ReadPrimitiveFromSource(m_reader); + return (*this); + } SinkReaderStream & operator >> (bool & t) { diff --git a/data/liechtenstein.osm.bz2 b/data/liechtenstein.osm.bz2 deleted file mode 100644 index b4ac85807e..0000000000 Binary files a/data/liechtenstein.osm.bz2 and /dev/null differ diff --git a/geometry/cellid.hpp b/geometry/cellid.hpp index d58412d1cd..7d4fb0f736 100644 --- a/geometry/cellid.hpp +++ b/geometry/cellid.hpp @@ -54,6 +54,19 @@ public: return CellId(bits, level); } + static bool IsCellId(string const & s) + { + size_t const length = s.size(); + if (length == 0) + return false; + for (size_t i = 0; i < length; ++i) + { + if (s[i] < '0' || s[i] > '3') + return false; + } + return true; + } + static CellId FromString(string const & s) { int64_t bits = 0; diff --git a/geometry/geometry_tests/cellid_test.cpp b/geometry/geometry_tests/cellid_test.cpp index 7580d8d9b2..6753121c90 100644 --- a/geometry/geometry_tests/cellid_test.cpp +++ b/geometry/geometry_tests/cellid_test.cpp @@ -175,3 +175,13 @@ UNIT_TEST(CellID_LessStackOrder) TEST_EQUAL(tst, exp, ()); } while (next_permutation(e.begin(), e.end())); } + +UNIT_TEST(CellID_IsStringValid) +{ + typedef m2::CellId<9> TId; + TEST( TId::IsCellId("0123132"), () ); + TEST( !TId::IsCellId(""), () ); + TEST( !TId::IsCellId("-1332"), () ); + TEST( !TId::IsCellId("023."), () ); + TEST( !TId::IsCellId("121832"), () ); +} diff --git a/indexer/feature.hpp b/indexer/feature.hpp index 2de8e75e19..951151b25b 100644 --- a/indexer/feature.hpp +++ b/indexer/feature.hpp @@ -2,6 +2,8 @@ #include "cell_id.hpp" +#include "../storage/defines.hpp" // just for file extensions + #include "../geometry/point2d.hpp" #include "../geometry/rect2d.hpp" @@ -340,7 +342,7 @@ public: FileReader m_trgF; read_source_t(string const & name) - : m_gF(name + ".geom"), m_trgF(name + ".trg") + : m_gF(name + GEOMETRY_FILE_EXTENSION), m_trgF(name + TRIANGLES_FILE_EXTENSION) { } }; diff --git a/indexer/indexer_tool/feature_generator.cpp b/indexer/indexer_tool/feature_generator.cpp index 4d5b1db53c..ff26d4246f 100644 --- a/indexer/indexer_tool/feature_generator.cpp +++ b/indexer/indexer_tool/feature_generator.cpp @@ -3,6 +3,8 @@ #include "data_cache_file.hpp" #include "osm_element.hpp" +#include "../../storage/defines.hpp" + #include "../../indexer/data_header.hpp" #include "../../indexer/osm_decl.hpp" #include "../../indexer/data_header_reader.hpp" @@ -140,7 +142,7 @@ void FeaturesCollector::Init() } FeaturesCollector::FeaturesCollector(string const & fName) -: m_datFile(fName), m_geoFile(fName + ".geom"), m_trgFile(fName + ".trg") +: m_datFile(fName), m_geoFile(fName + GEOMETRY_FILE_EXTENSION), m_trgFile(fName + TRIANGLES_FILE_EXTENSION) { Init(); } @@ -148,8 +150,8 @@ FeaturesCollector::FeaturesCollector(string const & fName) FeaturesCollector::FeaturesCollector(string const & bucket, FeaturesCollector::InitDataType const & prefix) : m_datFile(prefix.first + bucket + prefix.second), - m_geoFile(prefix.first + bucket + prefix.second + ".geom"), - m_trgFile(prefix.first + bucket + prefix.second + ".trg") + m_geoFile(prefix.first + bucket + prefix.second + GEOMETRY_FILE_EXTENSION), + m_trgFile(prefix.first + bucket + prefix.second + TRIANGLES_FILE_EXTENSION) { Init(); } diff --git a/indexer/indexer_tool/indexer_tool.cpp b/indexer/indexer_tool/indexer_tool.cpp index 4f63d0e94c..c4d7ee0b02 100644 --- a/indexer/indexer_tool/indexer_tool.cpp +++ b/indexer/indexer_tool/indexer_tool.cpp @@ -27,8 +27,8 @@ #include "../../base/start_mem_debug.hpp" DEFINE_bool(version, false, "Display version"); -DEFINE_string(countries_path, "", - "If specified, update.maps file will be generated from cells in the path"); +DEFINE_bool(generate_update, false, + "If specified, update.maps file will be generated from cells in the data path"); DEFINE_bool(sort_features, false, "Sort features inside .dat for better cache-friendliness."); DEFINE_bool(generate_classif, false, "Generate classificator."); @@ -141,8 +141,7 @@ int main(int argc, char ** argv) if (FLAGS_generate_index) { LOG(LINFO, ("Generating index for", datFile)); - string const indexFile = storage::IndexFileForDatFile(datFile); - if (!indexer::BuildIndexFromDatFile(indexFile, datFile, + if (!indexer::BuildIndexFromDatFile(datFile + INDEX_FILE_EXTENSION, datFile, FLAGS_intermediate_data_path + FLAGS_output)) { LOG(LCRITICAL, ("Error generating index.")); @@ -151,10 +150,10 @@ int main(int argc, char ** argv) } // Create http update list for countries and corresponding files - if (FLAGS_countries_path.size()) + if (FLAGS_generate_update) { LOG(LINFO, ("Creating maps.update file...")); - update::GenerateMapsList(path, FLAGS_countries_path, UPDATE_BASE_URL); + update::GenerateFilesList(path); } return 0; diff --git a/indexer/indexer_tool/update_generator.cpp b/indexer/indexer_tool/update_generator.cpp index 57051b52ee..f66b6802e0 100644 --- a/indexer/indexer_tool/update_generator.cpp +++ b/indexer/indexer_tool/update_generator.cpp @@ -2,6 +2,8 @@ #include "../../coding/file_writer.hpp" +#include "../../geometry/cellid.hpp" + #include "../../platform/platform.hpp" #include "../../storage/country.hpp" @@ -13,207 +15,81 @@ #include "../../std/target_os.hpp" #include "../../std/fstream.hpp" -#define GROUP_FILE_EXTENSION ".group" -#define REGION_FILE_EXTENSION ".regions" -#define CELLID_FILE_EXTENSION ".cells" - -#define PREFIX_CHAR '-' - -#ifdef OMIM_OS_WINDOWS_NATIVE - #define DIR_SEP '\\' -#else - #define DIR_SEP '/' -#endif - using namespace storage; namespace update { - typedef vector TCellIds; - typedef pair TCountryCells; - typedef vector TCountryCellsContainer; - - string ChopExtension(string const & nameWithExtension) + // we don't support files without name or without extension + bool SplitExtension(string const & file, string & name, string & ext) { - size_t dotPos = nameWithExtension.rfind('.'); - return nameWithExtension.substr(0, dotPos); - } - - string ChopPrefix(string const & nameWithPrefix) - { - size_t prefixPos = nameWithPrefix.rfind(PREFIX_CHAR); - if (prefixPos != string::npos && (prefixPos + 1) < nameWithPrefix.size()) - return nameWithPrefix.substr(prefixPos + 1, string::npos); - else - return nameWithPrefix; - } - - bool LoadCountryCells(string const & fullPath, TCellIds & outCells) - { - outCells.clear(); - ifstream file(fullPath.c_str()); - string cell; - while (file.good()) + // get extension + size_t const index = file.find_last_of('.'); + if (index == string::npos || (index + 1) == file.size() || index == 0 + || file == "." || file == "..") { - getline(file, cell); - if (cell.size()) - outCells.push_back(cell); - } - return !outCells.empty(); - } - - /// @param path where folders with groups reside (Africa, Asia etc.) - /// @return true if loaded correctly - bool ScanAndLoadCountryCells(string path, TCountryCellsContainer & outCells) - { - if (path.empty()) + name = file; + ext.clear(); return false; - // fix missing slash - if (path[path.size() - 1] != DIR_SEP) - path.push_back(DIR_SEP); - - outCells.clear(); + } + ext = file.substr(index); + name = file.substr(0, index); + return true; + } + bool GenerateFilesList(string const & dataDir) + { Platform & platform = GetPlatform(); - // get all groups - Platform::FilesList groups; - platform.GetFilesInDir(path, "*" GROUP_FILE_EXTENSION, groups); - for (Platform::FilesList::iterator itGroup = groups.begin(); itGroup != groups.end(); ++itGroup) + + Platform::FilesList files; + if (!platform.GetFilesInDir(dataDir, "*", files)) { - // get countries with regions - Platform::FilesList countries; - platform.GetFilesInDir(path + DIR_SEP + *itGroup, "*" REGION_FILE_EXTENSION, countries); - for (Platform::FilesList::iterator itCountry = countries.begin(); itCountry != countries.end(); ++itCountry) - { // get all regions - Platform::FilesList regions; - platform.GetFilesInDir(path + DIR_SEP + *itGroup + DIR_SEP + *itCountry, - "*" CELLID_FILE_EXTENSION, regions); - for (Platform::FilesList::iterator itRegion = regions.begin(); itRegion != regions.end(); ++itRegion) + LOG(LERROR, ("Can't find any files at path", dataDir)); + return false; + } + + TDataFiles cellFiles; + TCommonFiles commonFiles; + string name, ext; + int32_t level = -1; + uint16_t bits; + for (Platform::FilesList::iterator it = files.begin(); it != files.end(); ++it) + { + uint64_t size = 0; + CHECK( platform.GetFileSize(dataDir + *it, size), ()); + CHECK_EQUAL( size, static_cast(size), ("We don't support files > 4gb", *it)); + if (SplitExtension(*it, name, ext)) + { + // is it data cell file? + if (ext == DATA_FILE_EXTENSION) { - TCellIds cells; - string fullPath = path + *itGroup + DIR_SEP + *itCountry + DIR_SEP + *itRegion; - if (LoadCountryCells(fullPath, cells)) + if (CountryCellId::IsCellId(name)) { - outCells.push_back(TCountryCells(Country(ChopExtension(*itGroup), - ChopExtension(*itCountry), - ChopExtension(*itRegion)), - cells)); + CountryCellId cellId = CountryCellId::FromString(name); + pair bl = cellId.ToBitsAndLevel(); + if (level < 0) + level = bl.second; + CHECK_EQUAL( level, bl.second, ("Data files with different level?", *it) ); + bits = static_cast(bl.first); + CHECK_EQUAL( name, CountryCellId::FromBitsAndLevel(bits, level).ToString(), (name)); + cellFiles.push_back(make_pair(bits, static_cast(size))); } else { - LOG(LERROR, ("Can't load cells from file", fullPath)); + commonFiles.push_back(make_pair(*it, static_cast(size))); } } - } - - // get countries without regions - countries.clear(); - platform.GetFilesInDir(path + DIR_SEP + *itGroup, "*" CELLID_FILE_EXTENSION, countries); - for (Platform::FilesList::iterator itCountry = countries.begin(); itCountry != countries.end(); ++itCountry) - { - TCellIds cells; - string fullPath = path + *itGroup + DIR_SEP + *itCountry; - if (LoadCountryCells(fullPath, cells)) - { - outCells.push_back(TCountryCells(Country(ChopExtension(*itGroup), - ChopExtension(*itCountry), - ""), - cells)); - } else { - LOG(LERROR, ("Can't load cells from file", fullPath)); + commonFiles.push_back(make_pair(*it, static_cast(size))); } } } - return !outCells.empty(); - } - class CellChecker - { - string const m_path, m_file, m_url; + SaveTiles(dataDir + UPDATE_CHECK_FILE, level, cellFiles, commonFiles); - public: - CellChecker(string const & dataPath, string const & dataFileName, string const & baseUrl) - : m_path(dataPath), m_file(dataFileName), m_url(baseUrl) {} - void operator()(TCountryCells & cells) const - { - string const fileCell = ChopPrefix(ChopExtension(m_file)); - for (TCellIds::iterator it = cells.second.begin(); it != cells.second.end(); ++it) - { - // check if country contains tile with this cell - if (fileCell.find(*it) == 0 || it->find(fileCell) == 0) - { - // data file - uint64_t fileSize = 0; - CHECK(GetPlatform().GetFileSize(m_path + m_file, fileSize), ("Non-existing file?")); - cells.first.AddUrl(TUrl(m_url + m_file, fileSize)); - // index file - string const indexFileName = IndexFileForDatFile(m_file); - CHECK(GetPlatform().GetFileSize(m_path + indexFileName, fileSize), ("Non-existing file?")); - cells.first.AddUrl(TUrl(m_url + indexFileName, fileSize)); - break; - } - } - } - }; + LOG_SHORT(LINFO, ("Created update file with", cellFiles.size(), "data files and", + commonFiles.size(), "other files")); - class CountryAdder - { - TCountriesContainer & m_countries; - - public: - CountryAdder(TCountriesContainer & outCountries) - : m_countries(outCountries) {} - void operator()(TCountryCells const & cells) - { - if (cells.first.Urls().size()) - m_countries[cells.first.Group()].push_back(cells.first); - } - }; - - class GroupSorter - { - public: - void operator()(TCountriesContainer::value_type & toSort) - { - sort(toSort.second.begin(), toSort.second.end()); - } - }; - - bool GenerateMapsList(string const & pathToMaps, string const & pathToCountries, string const & baseUrl) - { - Platform & platform = GetPlatform(); - - TCountryCellsContainer countriesCells; - if (!ScanAndLoadCountryCells(pathToCountries, countriesCells)) - { - LOG(LERROR, ("Can't load countries' cells from path", pathToCountries)); - return false; - } - - Platform::FilesList datFiles; - if (!platform.GetFilesInDir(pathToMaps, "*" DATA_FILE_EXTENSION, datFiles)) - { - LOG(LERROR, ("Can't find any data files at path", pathToMaps)); - return false; - } - - // update each country's urls corresponding to existing data files - for (Platform::FilesList::iterator it = datFiles.begin(); it != datFiles.end(); ++it) - { - for_each(countriesCells.begin(), countriesCells.end(), CellChecker(pathToMaps, *it, baseUrl)); - } - - // save update list - TCountriesContainer countries; - for_each(countriesCells.begin(), countriesCells.end(), CountryAdder(countries)); - - // sort groups - for_each(countries.begin(), countries.end(), GroupSorter()); - - FileWriter writer(pathToMaps + UPDATE_CHECK_FILE); - SaveCountries(countries, writer); return true; } } // namespace update diff --git a/indexer/indexer_tool/update_generator.hpp b/indexer/indexer_tool/update_generator.hpp index 5ae5a590e6..bc7a0f8c63 100644 --- a/indexer/indexer_tool/update_generator.hpp +++ b/indexer/indexer_tool/update_generator.hpp @@ -4,5 +4,5 @@ namespace update { - bool GenerateMapsList(string const & pathToMaps, string const & pathToCountries, string const & baseUrl); + bool GenerateFilesList(string const & dataDir); } // namespace update diff --git a/qt/update_dialog.cpp b/qt/update_dialog.cpp index 70ec013699..f2702b5ec7 100644 --- a/qt/update_dialog.cpp +++ b/qt/update_dialog.cpp @@ -1,10 +1,12 @@ #include "update_dialog.hpp" +#include "../base/assert.hpp" + #include #include #include -#include +#include #include #include #include @@ -14,9 +16,9 @@ using namespace storage; enum { // KItemIndexFlag = 0, - KItemIndexCountry, - KItemIndexContinent, - KItemIndexSize, + KColumnIndexCountry, + KColumnIndexStatus, + KColumnIndexSize, KNumberOfColumns }; @@ -32,12 +34,12 @@ namespace qt // Helpers /////////////////////////////////////////////////////////////////////////////// /// adds custom sorting for "Size" column - class QTableWidgetItemWithCustomSorting : public QTableWidgetItem + class QTreeWidgetItemWithCustomSorting : public QTreeWidgetItem { public: - virtual bool operator<(QTableWidgetItem const & other) const + virtual bool operator<(QTreeWidgetItem const & other) const { - return data(Qt::UserRole).toULongLong() < other.data(Qt::UserRole).toULongLong(); + return data(KColumnIndexSize, Qt::UserRole).toULongLong() < other.data(KColumnIndexSize, Qt::UserRole).toULongLong(); } }; //////////////////////////////////////////////////////////////////////////////// @@ -45,28 +47,25 @@ namespace qt UpdateDialog::UpdateDialog(QWidget * parent, Storage & storage) : QDialog(parent), m_storage(storage) { - // table with countries list - m_table = new QTableWidget(0, KNumberOfColumns, this); - m_table->setSelectionBehavior(QAbstractItemView::SelectRows); - QStringList labels; - labels << tr("Country") << tr("Continent") << tr("Size"); - m_table->setHorizontalHeaderLabels(labels); - m_table->horizontalHeader()->setResizeMode(KItemIndexCountry, QHeaderView::Stretch); - m_table->verticalHeader()->hide(); - m_table->setShowGrid(false); - connect(m_table, SIGNAL(cellClicked(int, int)), this, SLOT(OnTableClick(int, int))); + m_tree = new QTreeWidget(this); + m_tree->setColumnCount(KNumberOfColumns); + QStringList columnLabels; + columnLabels << tr("Country") << tr("Status") << tr("Size"); + m_tree->setHeaderLabels(columnLabels); + + connect(m_tree, SIGNAL(itemClicked(QTreeWidgetItem *, int)), this, SLOT(OnItemClick(QTreeWidgetItem *, int))); QVBoxLayout * layout = new QVBoxLayout(); - layout->addWidget(m_table); + layout->addWidget(m_tree); setLayout(layout); - setWindowTitle(tr("Countries")); - resize(500, 300); + setWindowTitle(tr("Geographical Regions")); + resize(600, 500); // we want to receive all download progress and result events m_storage.Subscribe(boost::bind(&UpdateDialog::OnCountryChanged, this, _1), boost::bind(&UpdateDialog::OnCountryDownloadProgress, this, _1, _2)); - FillTable(); + FillTree(); } UpdateDialog::~UpdateDialog() @@ -76,137 +75,172 @@ namespace qt } /// when user clicks on any map row in the table - void UpdateDialog::OnTableClick(int row, int /*column*/) + void UpdateDialog::OnItemClick(QTreeWidgetItem * item, int /*column*/) { - uint group = m_table->item(row, KItemIndexContinent)->data(Qt::UserRole).toUInt(); - uint country = m_table->item(row, KItemIndexCountry)->data(Qt::UserRole).toUInt(); + // calculate index of clicked item + QList treeIndex; + { + QTreeWidgetItem * parent = item; + while (parent) + { + treeIndex.insert(0, parent->data(KColumnIndexCountry, Qt::UserRole).toInt()); + parent = parent->parent(); + } + while (treeIndex.size() < 3) + treeIndex.append(-1); + } - switch (m_storage.CountryStatus(TIndex(group, country))) + TIndex const countryIndex(treeIndex[0], treeIndex[1], treeIndex[2]); + switch (m_storage.CountryStatus(countryIndex)) { case EOnDisk: { // aha.. map is already downloaded, so ask user about deleting! QMessageBox ask(this); - ask.setText(tr("Do you want to delete %1?").arg(m_table->item(row, KItemIndexCountry)->text())); + ask.setText(tr("Do you want to delete %1?").arg(item->text(KColumnIndexCountry))); ask.setStandardButtons(QMessageBox::Yes | QMessageBox::No); ask.setDefaultButton(QMessageBox::No); if (ask.exec() == QMessageBox::Yes) - m_storage.DeleteCountry(TIndex(group, country)); + m_storage.DeleteCountry(countryIndex); } break; case ENotDownloaded: case EDownloadFailed: - m_storage.DownloadCountry(TIndex(group, country)); + m_storage.DownloadCountry(countryIndex); break; case EInQueue: case EDownloading: - m_storage.DeleteCountry(TIndex(group, country)); + m_storage.DeleteCountry(countryIndex); break; } } - int GetRowByGroupAndCountryIndex(QTableWidget & table, TIndex const & index) + /// @return can be null if index is invalid + QTreeWidgetItem * GetTreeItemByIndex(QTreeWidget & tree, TIndex const & index) { - for (int i = 0; i < table.rowCount(); ++i) + QTreeWidgetItem * item = 0; + if (index.m_group >= 0 && index.m_group < tree.topLevelItemCount()) { - if (table.item(i, KItemIndexContinent)->data(Qt::UserRole).toUInt() == index.first - && table.item(i, KItemIndexCountry)->data(Qt::UserRole).toUInt() == index.second) + item = tree.topLevelItem(index.m_group); + ASSERT_EQUAL( item->data(KColumnIndexCountry, Qt::UserRole).toInt(), index.m_group, () ); + if (index.m_country >= 0 && index.m_country < item->childCount()) { - return i; + item = item->child(index.m_country); + ASSERT_EQUAL( item->data(KColumnIndexCountry, Qt::UserRole).toInt(), index.m_country, () ); + if (index.m_region >= 0 && index.m_region < item->childCount()) + { + item = item->child(index.m_region); + ASSERT_EQUAL( item->data(KColumnIndexCountry, Qt::UserRole).toInt(), index.m_region, () ); + } } } - // not found - return -1; + return item; } /// Changes row's text color - void SetRowColor(QTableWidget & table, int row, QColor const & color) + void SetRowColor(QTreeWidgetItem & item, QColor const & color) { - for (int column = 0; column < table.columnCount(); ++column) - { - QTableWidgetItem * item = table.item(row, column); - if (item) - item->setTextColor(color); - } + for (int column = 0; column < item.columnCount(); ++column) + item.setTextColor(column, color); } void UpdateDialog::UpdateRowWithCountryInfo(TIndex const & index) { - int row = GetRowByGroupAndCountryIndex(*m_table, index); - if (row == -1) + QTreeWidgetItem * item = GetTreeItemByIndex(*m_tree, index); + if (item) { - ASSERT(false, ("Invalid country index")); - return; - } + QColor rowColor; + QString statusString; + switch (m_storage.CountryStatus(index)) + { + case ENotDownloaded: + statusString = tr("Click to download"); + rowColor = COLOR_NOTDOWNLOADED; + break; + case EOnDisk: + statusString = tr("Installed (click to delete)"); + rowColor = COLOR_ONDISK; + break; + case EDownloadFailed: + statusString = tr("Download has failed :("); + rowColor = COLOR_DOWNLOADFAILED; + break; + case EDownloading: + statusString = tr("Downloading..."); + rowColor = COLOR_INPROGRESS; + break; + case EInQueue: + statusString = tr("Marked for download"); + rowColor = COLOR_INQUEUE; + break; + } + item->setText(KColumnIndexStatus, statusString); - QColor rowColor; - TStatus status = m_storage.CountryStatus(index); - switch (status) - { - case ENotDownloaded: rowColor = COLOR_NOTDOWNLOADED; break; - case EOnDisk: rowColor = COLOR_ONDISK; break; - case EDownloadFailed: rowColor = COLOR_DOWNLOADFAILED; break; - case EDownloading: rowColor = COLOR_INPROGRESS; break; - case EInQueue: rowColor = COLOR_INQUEUE; break; + // update size column values +// QTableWidgetItem * sizeItem = m_table->item(row, KColumnIndexSize); +// if (status == EInQueue) +// { +// sizeItem->setText("In Queue"); +// } +// else if (status != EDownloading) +// { +// uint64_t const size = m_storage.CountrySizeInBytes(index); +// if (size > 1000 * 1000) +// sizeItem->setText(QObject::tr("%1 MB").arg(uint(size / (1000 * 1000)))); +// else +// sizeItem->setText(QObject::tr("%1 kB").arg(uint((size + 999) / 1000))); +// // needed for column sorting +// sizeItem->setData(Qt::UserRole, QVariant(qint64(size))); +// } + SetRowColor(*item, rowColor); } - - // update size column values - QTableWidgetItem * sizeItem = m_table->item(row, KItemIndexSize); - if (status == EInQueue) - { - sizeItem->setText("In Queue"); - } - else if (status != EDownloading) - { - uint64_t const size = m_storage.CountrySizeInBytes(index); - if (size > 1000 * 1000) - sizeItem->setText(QObject::tr("%1 MB").arg(uint(size / (1000 * 1000)))); - else - sizeItem->setText(QObject::tr("%1 kB").arg(uint((size + 999) / 1000))); - // needed for column sorting - sizeItem->setData(Qt::UserRole, QVariant(qint64(size))); - } - SetRowColor(*m_table, row, rowColor); } - void UpdateDialog::FillTable() + void UpdateDialog::FillTree() { - m_table->setSortingEnabled(false); - m_table->clear(); + m_tree->setSortingEnabled(false); + m_tree->clear(); - for (size_t groupIndex = 0; groupIndex < m_storage.GroupsCount(); ++groupIndex) + for (int group = 0; group < m_storage.CountriesCount(TIndex()); ++group) { - for (size_t countryIndex = 0; countryIndex < m_storage.CountriesCountInGroup(groupIndex); ++countryIndex) + TIndex const grIndex(group); + QStringList groupText(m_storage.CountryName(grIndex).c_str()); + QTreeWidgetItem * groupItem = new QTreeWidgetItem(groupText); + groupItem->setData(KColumnIndexCountry, Qt::UserRole, QVariant(group)); + m_tree->addTopLevelItem(groupItem); + // set color by status and update country size + UpdateRowWithCountryInfo(grIndex); + + for (int country = 0; country < m_storage.CountriesCount(grIndex); ++country) { - int row = m_table->rowCount(); - m_table->insertRow(row); - - // Country column - QTableWidgetItem * countryItem = new QTableWidgetItem(m_storage.CountryName(TIndex(groupIndex, countryIndex)).c_str()); - countryItem->setFlags(countryItem->flags() ^ Qt::ItemIsEditable); - // acts as a key for Storage when user clicks this row - countryItem->setData(Qt::UserRole, QVariant(uint(countryIndex))); - m_table->setItem(row, KItemIndexCountry, countryItem); - - // Continent column - QTableWidgetItem * continentItem = new QTableWidgetItem(m_storage.GroupName(groupIndex).c_str()); - continentItem->setFlags(continentItem->flags() ^ Qt::ItemIsEditable); - // acts as a key for Storage when user clicks this row - continentItem->setData(Qt::UserRole, QVariant(uint(groupIndex))); - m_table->setItem(row, KItemIndexContinent, continentItem); - - // Size column, actual size will be set later - QTableWidgetItemWithCustomSorting * sizeItem = new QTableWidgetItemWithCustomSorting; - sizeItem->setFlags(sizeItem->flags() ^ Qt::ItemIsEditable); - m_table->setItem(row, KItemIndexSize, sizeItem); - + TIndex cIndex(group, country); + QStringList countryText(m_storage.CountryName(cIndex).c_str()); + QTreeWidgetItem * countryItem = new QTreeWidgetItem(groupItem, countryText); + countryItem->setData(KColumnIndexCountry, Qt::UserRole, QVariant(country)); // set color by status and update country size - UpdateRowWithCountryInfo(TIndex(groupIndex, countryIndex)); + UpdateRowWithCountryInfo(cIndex); + + for (int region = 0; region < m_storage.CountriesCount(cIndex); ++region) + { + TIndex const rIndex(group, country, region); + QStringList regionText(m_storage.CountryName(rIndex).c_str()); + QTreeWidgetItem * regionItem = new QTreeWidgetItem(countryItem, regionText); + regionItem->setData(KColumnIndexCountry, Qt::UserRole, QVariant(region)); + // set color by status and update country size + UpdateRowWithCountryInfo(rIndex); + } } } - m_table->setSortingEnabled(true); - m_table->sortByColumn(KItemIndexCountry); +// // Size column, actual size will be set later +// QTableWidgetItemWithCustomSorting * sizeItem = new QTableWidgetItemWithCustomSorting; +// sizeItem->setFlags(sizeItem->flags() ^ Qt::ItemIsEditable); +// m_table->setItem(row, KItemIndexSize, sizeItem); + + m_tree->sortByColumn(KColumnIndexCountry, Qt::AscendingOrder); + m_tree->setSortingEnabled(true); + m_tree->header()->setResizeMode(KColumnIndexCountry, QHeaderView::ResizeToContents); + m_tree->header()->setResizeMode(KColumnIndexStatus, QHeaderView::ResizeToContents); } void UpdateDialog::OnCountryChanged(TIndex const & index) @@ -214,18 +248,12 @@ namespace qt UpdateRowWithCountryInfo(index); } - void UpdateDialog::OnCountryDownloadProgress(TIndex const & index, TDownloadProgress const & progress) + void UpdateDialog::OnCountryDownloadProgress(TIndex const & index, + TDownloadProgress const & progress) { - int row = GetRowByGroupAndCountryIndex(*m_table, index); - if (row != -1) - { - m_table->item(row, KItemIndexSize)->setText( - QString("%1%").arg(progress.first * 100 / progress.second)); - } - else - { - ASSERT(false, ()); - } + QTreeWidgetItem * item = GetTreeItemByIndex(*m_tree, index); + if (item) + item->setText(KColumnIndexSize, QString("%1%").arg(progress.first * 100 / progress.second)); } } diff --git a/qt/update_dialog.hpp b/qt/update_dialog.hpp index dfdc2244a5..7dcf2bff85 100644 --- a/qt/update_dialog.hpp +++ b/qt/update_dialog.hpp @@ -4,7 +4,8 @@ #include -class QTableWidget; +class QTreeWidget; +class QTreeWidgetItem; class QLabel; namespace qt @@ -17,7 +18,7 @@ namespace qt UpdateDialog(QWidget * parent, storage::Storage & storage); ~UpdateDialog(); - /// @name Called from storage to notify GUI + /// @name Called from downloader to notify GUI //@{ void OnCountryChanged(storage::TIndex const & index); void OnCountryDownloadProgress(storage::TIndex const & index, @@ -25,14 +26,14 @@ namespace qt //@} private slots: - void OnTableClick(int row, int column); + void OnItemClick(QTreeWidgetItem * item, int column); private: - void FillTable(); + void FillTree(); void UpdateRowWithCountryInfo(storage::TIndex const & index); private: - QTableWidget * m_table; + QTreeWidget * m_tree; storage::Storage & m_storage; }; } // namespace qt diff --git a/storage/country.cpp b/storage/country.cpp index f63f1fef11..8762cd9126 100644 --- a/storage/country.cpp +++ b/storage/country.cpp @@ -2,6 +2,10 @@ #include "../base/logging.hpp" + +#include "../base/std_serialization.hpp" + +#include "../coding/streams_sink.hpp" #include "../coding/file_reader.hpp" #include "../coding/file_writer.hpp" @@ -10,135 +14,194 @@ #include "../indexer/data_header.hpp" #include "../indexer/data_header_reader.hpp" +#include "../std/fstream.hpp" + namespace storage { - string FileNameFromUrl(string const & url) - { - size_t lastSlashPos = url.find_last_of('/'); - if (lastSlashPos != string::npos && (lastSlashPos + 1) < url.size()) - return url.substr(lastSlashPos + 1); - ASSERT( false, ("Url should be valid") ); - return string(); - } - - bool IsFileSizeEqualTo(string const & fileName, uint64_t size) - { - uint64_t diskSize = 0; - if (GetPlatform().GetFileSize(GetPlatform().WritablePathForFile(fileName), diskSize) - && diskSize == size) - return true; - else - return false; - } - /// Simple check - compare url size with real file size on disk - bool IsFileDownloaded(TUrl const & url) + bool IsTileDownloaded(TTile const & tile) { - string fileName = FileNameFromUrl(url.first); - return IsFileSizeEqualTo(fileName, url.second); + uint64_t size = 0; + if (!GetPlatform().GetFileSize(GetPlatform().WritablePathForFile(tile.first), size)) + return false; + return true;//tile.second == size; } struct CountryBoundsCalculator { m2::RectD & m_bounds; CountryBoundsCalculator(m2::RectD & bounds) : m_bounds(bounds) {} - void operator()(TUrl const & url) + void operator()(TTile const & tile) { - string fileName = FileNameFromUrl(url.first); - if (IsFileSizeEqualTo(fileName, url.second) && IsDatFile(fileName)) - { - feature::DataHeader header; - if (feature::ReadDataHeader(GetPlatform().WritablePathForFile(fileName), header)) - m_bounds.Add(header.Bounds()); - } + static feature::DataHeader header; + if (feature::ReadDataHeader(GetPlatform().WritablePathForFile(tile.first), header)) + m_bounds.Add(header.Bounds()); } }; m2::RectD Country::Bounds() const { m2::RectD bounds; - std::for_each(m_urls.begin(), m_urls.end(), CountryBoundsCalculator(bounds)); + std::for_each(m_tiles.begin(), m_tiles.end(), CountryBoundsCalculator(bounds)); return bounds; } - struct LocalSizeCalculator + struct SizeCalculator { - uint64_t & m_size; - LocalSizeCalculator(uint64_t & size) : m_size(size) {} - void operator()(TUrl const & url) + uint64_t & m_localSize; + uint64_t & m_remoteSize; + SizeCalculator(uint64_t & localSize, uint64_t & remoteSize) + : m_localSize(localSize), m_remoteSize(remoteSize) {} + void operator()(TTile const & tile) { - if (IsFileDownloaded(url)) - m_size += url.second; + if (IsTileDownloaded(tile)) + m_localSize += tile.second; + else + m_remoteSize += tile.second; } }; - uint64_t Country::LocalSize() const + TLocalAndRemoteSize Country::Size() const { - uint64_t size = 0; - std::for_each(m_urls.begin(), m_urls.end(), LocalSizeCalculator(size)); - return size; + uint64_t localSize = 0; + uint64_t remoteSize = 0; + std::for_each(m_tiles.begin(), m_tiles.end(), SizeCalculator(localSize, remoteSize)); + return TLocalAndRemoteSize(localSize, remoteSize); } - struct RemoteSizeCalculator + void Country::AddTile(TTile const & tile) { - uint64_t & m_size; - RemoteSizeCalculator(uint64_t & size) : m_size(size) {} - void operator()(TUrl const & url) - { - if (!IsFileDownloaded(url)) - m_size += url.second; - } - }; - - uint64_t Country::RemoteSize() const - { - uint64_t size = 0; - std::for_each(m_urls.begin(), m_urls.end(), RemoteSizeCalculator(size)); - return size; - } - - void Country::AddUrl(TUrl const & url) - { - m_urls.push_back(url); + m_tiles.push_back(tile); } //////////////////////////////////////////////////////////////////////// - template TArchive & operator << (TArchive & ar, storage::Country const & country) +// template TArchive & operator << (TArchive & ar, storage::Country const & country) +// { +// ar << country.m_name; +// ar << country.m_tiles; +// return ar; +// } + + inline bool IsCellId(string const & cellId) { - ar << country.m_group; - ar << country.m_country; - ar << country.m_region; - ar << country.m_urls; - return ar; + size_t const size = cellId.size(); + if (size == 0) + return false; + for (size_t i = 0; i < size; ++i) + { + if (cellId[i] < '0' || cellId[i] > '3') + return false; + } + return true; } - bool LoadCountries(TCountriesContainer & countries, string const & updateFile) + bool LoadCountries(string const & countriesFile, TTilesContainer const & sortedTiles, + TCountriesContainer & countries) { - countries.clear(); + countries.Clear(); + ifstream stream(countriesFile.c_str()); + std::string line; + Country * currentCountry = &countries.Value(); + while (stream.good()) + { + std::getline(stream, line); + if (line.empty()) + continue; + + // calculate spaces - depth inside the tree + int spaces = 0; + for (size_t i = 0; i < line.size(); ++i) + { + if (line[i] == ' ') + ++spaces; + else + break; + } + switch (spaces) + { + case 0: // this is value for current tree node + { + TTilesContainer::const_iterator const first = sortedTiles.begin(); + TTilesContainer::const_iterator const last = sortedTiles.end(); + TTilesContainer::const_iterator found = lower_bound(first, last, TTile(line, 0)); + // @TODO current implementation supports only cellId level 8 and 7 + if (!(found != last && !(line < found->first)) && IsCellId(line)) + { + line.resize(line.size() - 1); + found = lower_bound(first, last, TTile(line, 0)); + } + + if (found != last && !(line < found->first)) + currentCountry->AddTile(*found); + } + break; + case 1: // country group + case 2: // country name + case 3: // region + currentCountry = &countries.AddAtDepth(spaces - 1, Country(line.substr(spaces))); + break; + default: + return false; + } + } + return true; + } + + void SaveTiles(string const & file, int32_t level, TDataFiles const & cellFiles, TCommonFiles const & commonFiles) + { + FileWriter writer(file); + stream::SinkWriterStream wStream(writer); + wStream << MAPS_MAJOR_VERSION_BINARY_FORMAT; + wStream << level; + wStream << cellFiles; + wStream << commonFiles; + } + + bool LoadTiles(TTilesContainer & tiles, string const & tilesFile) + { + tiles.clear(); + try { - FileReader file(updateFile.c_str()); - ReaderSource source(file); - stream::SinkReaderStream > rStream(source); - uint32_t version; - rStream >> version; + FileReader fileReader(tilesFile); + ReaderSource source(fileReader); + stream::SinkReaderStream > stream(source); + + TDataFiles dataFiles; + TCommonFiles commonFiles; + + uint32_t version = 0; + int32_t level = -1; + stream >> version; if (version > MAPS_MAJOR_VERSION_BINARY_FORMAT) return false; - rStream >> countries; - return true; + stream >> level; + stream >> dataFiles; + stream >> commonFiles; + + tiles.reserve(dataFiles.size() + commonFiles.size()); + + for (TDataFiles::iterator it = dataFiles.begin(); it != dataFiles.end(); ++it) + tiles.push_back(TTile(CountryCellId::FromBitsAndLevel(it->first, level).ToString(), it->second)); + for (TCommonFiles::iterator it = commonFiles.begin(); it != commonFiles.end(); ++it) + tiles.push_back(TTile(it->first, it->second)); + + sort(tiles.begin(), tiles.end()); } catch (RootException const & e) { - LOG(LERROR, ("LoadCountries exception", e.what())); + LOG(LWARNING, ("Can't read tiles file", e.what())); + return false; } - return false; + + return true; } - void SaveCountries(TCountriesContainer const & countries, Writer & writer) - { - stream::SinkWriterStream wStream(writer); - wStream << MAPS_MAJOR_VERSION_BINARY_FORMAT; - wStream << countries; - } +// void SaveCountries(TCountriesContainer const & countries, Writer & writer) +// { +// stream::SinkWriterStream wStream(writer); +// wStream << MAPS_MAJOR_VERSION_BINARY_FORMAT; +// wStream << countries; +// } } diff --git a/storage/country.hpp b/storage/country.hpp index 85fcf3dffe..9da9a99c7e 100644 --- a/storage/country.hpp +++ b/storage/country.hpp @@ -1,12 +1,10 @@ #pragma once #include "defines.hpp" - -#include "../base/std_serialization.hpp" - -#include "../coding/streams_sink.hpp" +#include "simple_tree.hpp" #include "../geometry/rect2d.hpp" +#include "../geometry/cellid.hpp" #include "../std/string.hpp" #include "../std/vector.hpp" @@ -16,70 +14,64 @@ class Writer; namespace storage { - typedef pair TUrl; - typedef vector TUrlContainer; + /// holds file name for tile and it's total size + typedef pair TTile; + typedef vector TTilesContainer; + typedef pair TLocalAndRemoteSize; - /// helper function - string FileNameFromUrl(string const & url); - bool IsFileDownloaded(TUrl const & url); - bool IsDatFile(string const & fileName); + /// @name Intermediate containers for data transfer + //@{ + typedef vector > TDataFiles; + typedef vector > TCommonFiles; + typedef m2::CellId<9> CountryCellId; + //@} + + bool IsTileDownloaded(TTile const & tile); /// Serves as a proxy between GUI and downloaded files class Country { - template friend TArchive & operator << (TArchive & ar, storage::Country const & country); - template friend TArchive & operator >> (TArchive & ar, storage::Country & country); +// template friend TArchive & operator << (TArchive & ar, storage::Country const & country); +// template friend TArchive & operator >> (TArchive & ar, storage::Country & country); private: - /// Europe, Asia etc. - string m_group; - /// USA, Switzerland etc. - string m_country; - /// can be empty, Alaska, California etc. - string m_region; - /// stores squares with world pieces where this country resides - TUrlContainer m_urls; + /// Name in the coutry node tree + string m_name; + /// stores squares with world pieces which are part of the country + TTilesContainer m_tiles; public: Country() {} - Country(string const & group, string const & country, string const & region) - : m_group(group), m_country(country), m_region(region) {} + Country(string const & name) + : m_name(name) {} bool operator<(Country const & other) const { return Name() < other.Name(); } - bool operator==(Country const & other) const - { - return m_group == other.m_group && m_country == other.m_country && m_region == other.m_region - && m_urls == other.m_urls; - } - bool operator!=(Country const & other) const { return !(*this == other); } - void AddUrl(TUrl const & url); - TUrlContainer const & Urls() const { return m_urls; } + void AddTile(TTile const & tile); + TTilesContainer const & Tiles() const { return m_tiles; } - string const & Group() const { return m_group; } - string Name() const { return m_region.empty() ? m_country : m_country + " - " + m_region; } + string Name() const { return m_name; } /// @return bounds for downloaded parts of the country or empty rect m2::RectD Bounds() const; - /// Downloaded parts size - uint64_t LocalSize() const; - /// Not downloaded parts size - uint64_t RemoteSize() const; + TLocalAndRemoteSize Size() const; }; - template TArchive & operator >> (TArchive & ar, storage::Country & country) - { - ar >> country.m_group; - ar >> country.m_country; - ar >> country.m_region; - ar >> country.m_urls; - return ar; - } +// template TArchive & operator >> (TArchive & ar, storage::Country & country) +// { +// ar >> country.m_name; +// ar >> country.m_tiles; +// return ar; +// } - /// key is Country::Group() - typedef map > TCountriesContainer; + typedef SimpleTree TCountriesContainer; + /// @param tiles contains files and their sizes /// @return false if new application version should be downloaded - bool LoadCountries(TCountriesContainer & countries, string const & updateFile); - void SaveCountries(TCountriesContainer const & countries, Writer & writer); + bool LoadCountries(string const & countriesFile, TTilesContainer const & sortedTiles, + TCountriesContainer & countries); + void SaveTiles(string const & file, int32_t level, TDataFiles const & cellFiles, + TCommonFiles const & commonFiles); + bool LoadTiles(TTilesContainer & tiles, string const & tilesFile); +// void SaveCountries(TCountriesContainer const & countries, Writer & writer); } diff --git a/storage/defines.hpp b/storage/defines.hpp index 7b5701c300..78a224f012 100644 --- a/storage/defines.hpp +++ b/storage/defines.hpp @@ -5,29 +5,16 @@ #include "../std/string.hpp" /// Should be incremented when binary format changes -uint32_t const MAPS_MAJOR_VERSION_BINARY_FORMAT = 0; +uint32_t const MAPS_MAJOR_VERSION_BINARY_FORMAT = 1; #define DATA_FILE_EXTENSION ".dat" +#define GEOMETRY_FILE_EXTENSION ".geom" +#define TRIANGLES_FILE_EXTENSION ".trg" +#define INDEX_FILE_EXTENSION ".idx" -#define WORLD_DATA_FILE "world" DATA_FILE_EXTENSION +//#define WORLD_DATA_FILE "world" DATA_FILE_EXTENSION +#define COUNTRIES_FILE "countries.txt" #define UPDATE_CHECK_FILE "maps.update" #define UPDATE_BASE_URL "http://melnichek.ath.cx:34568/maps/" #define UPDATE_FULL_URL UPDATE_BASE_URL UPDATE_CHECK_FILE - -namespace storage -{ - inline bool IsDatFile(string const & fileName) - { - /// file name ends with data file extension - string const ext(DATA_FILE_EXTENSION); - return fileName.rfind(ext) == fileName.size() - ext.size(); - } - - inline string IndexFileForDatFile(string const & fileName) - { - ASSERT(IsDatFile(fileName), ()); - static char const * INDEX_FILE_EXTENSION = ".idx"; - return fileName + INDEX_FILE_EXTENSION; - } -} diff --git a/storage/simple_tree.hpp b/storage/simple_tree.hpp index ae9c474ff8..c00b1b4998 100644 --- a/storage/simple_tree.hpp +++ b/storage/simple_tree.hpp @@ -93,8 +93,8 @@ public: { for (typename internal_container_type::iterator it = m_siblings.begin(); it != m_siblings.end(); ++it) { - it->ForEachChildren(f); f(*it); + it->ForEachChildren(f); } } @@ -103,8 +103,8 @@ public: { for (typename internal_container_type::const_iterator it = m_siblings.begin(); it != m_siblings.end(); ++it) { - it->ForEachChildren(f); f(*it); + it->ForEachChildren(f); } } }; diff --git a/storage/storage.cpp b/storage/storage.cpp index f9da514e7a..d7d4e3f279 100644 --- a/storage/storage.cpp +++ b/storage/storage.cpp @@ -14,171 +14,62 @@ namespace storage { - static bool IsMapValid(string const & datFile) - { - // @TODO add more serious integrity checks - string const indexFile = IndexFileForDatFile(datFile); - uint64_t datSize = 0, idxSize = 0; - bool result = GetPlatform().GetFileSize(datFile, datSize) - && GetPlatform().GetFileSize(indexFile, idxSize); - if (!result || datSize == 0 || idxSize == 0) - { - LOG(LINFO, ("Invalid map files:", datFile, indexFile)); - return false; - } - return true; - } - - /// Scan and load all existing local maps, calculate sum of maps' rects, - /// also deletes invalid or partially downloaded maps - template - class AddMapsToModelAndDeleteIfInvalid - { - TAddFn & m_addFunc; - size_t & m_addedCount; - public: - AddMapsToModelAndDeleteIfInvalid(TAddFn & addFunc, size_t & addedCount) - : m_addFunc(addFunc), m_addedCount(addedCount) {} - /// @return true if map is ok and false if it's deleted or doesn't exist - bool operator()(string const & datFile) - { - if (IsMapValid(datFile)) - { - m_addFunc(datFile, IndexFileForDatFile(datFile)); - ++m_addedCount; - return true; - } - else - { - // looks like map files are bad... delete them - FileWriter::DeleteFile(datFile); - FileWriter::DeleteFile(IndexFileForDatFile(datFile)); - return false; - } - } - }; - - /// Operates with map files pairs (currently dat and idx) - template - void ForEachMapInDir(string const & dir, TFunctor & func) - { - Platform::FilesList fileList; - GetPlatform().GetFilesInDir(dir, "*" DATA_FILE_EXTENSION, fileList); - std::for_each(fileList.begin(), fileList.end(), - boost::bind(&TFunctor::operator(), &func, - boost::bind(std::plus(), dir, _1))); - } - - template - void RescanAndAddDownloadedMaps(TAddFn & addFunc) - { - // scan and load all local maps - size_t addedMapsCount = 0; - AddMapsToModelAndDeleteIfInvalid functor(addFunc, addedMapsCount); - ForEachMapInDir(GetPlatform().WritableDir(), functor); - try - { - functor(GetPlatform().ReadPathForFile(WORLD_DATA_FILE)); - } - catch (FileAbsentException const &) - { - if (addedMapsCount == 0) - { - LOG(LWARNING, ("No data files were found. Model is empty.")); - } - } - } - void Storage::Init(TAddMapFunction addFunc, TRemoveMapFunction removeFunc) { m_addMap = addFunc; m_removeMap = removeFunc; - UpdateCheck(); - - RescanAndAddDownloadedMaps(addFunc); + // activate all downloaded maps + Platform::FilesList filesList; + string const dataPath = GetPlatform().WritableDir(); + GetPlatform().GetFilesInDir(dataPath, "*" DATA_FILE_EXTENSION, filesList); + for (Platform::FilesList::iterator it = filesList.begin(); it != filesList.end(); ++it) + m_addMap(dataPath + *it, dataPath + *it + INDEX_FILE_EXTENSION); } - bool Storage::UpdateCheck() - { - GetDownloadManager().DownloadFile( - UPDATE_FULL_URL, - (GetPlatform().WritablePathForFile(UPDATE_CHECK_FILE)).c_str(), - boost::bind(&Storage::OnUpdateDownloadFinished, this, _1, _2), - TDownloadProgressFunction(), false); - return true; - } +// bool Storage::UpdateCheck() +// { +// GetDownloadManager().DownloadFile( +// UPDATE_FULL_URL, +// (GetPlatform().WritablePathForFile(UPDATE_CHECK_FILE)).c_str(), +// boost::bind(&Storage::OnUpdateDownloadFinished, this, _1, _2), +// TDownloadProgressFunction(), false); +// return true; +// } - Country const * Storage::CountryByIndex(TIndex const & index) const + TCountriesContainer const & NodeFromIndex(TCountriesContainer const & root, TIndex const & index) { - TIndex::first_type i = 0; - for (TCountriesContainer::const_iterator itGroup = m_countries.begin(); - itGroup != m_countries.end(); ++itGroup, ++i) + // complex logic to avoid [] out_of_bounds exceptions + if (index.m_group == -1 || index.m_group >= root.SiblingsCount()) + return root; + else { - if (i == index.first) - { - if (itGroup->second.size() > index.second) - return &(itGroup->second[index.second]); - break; - } + if (index.m_country == -1 || index.m_country >= root[index.m_group].SiblingsCount()) + return root[index.m_group]; + if (index.m_region == -1 || index.m_region >= root[index.m_group][index.m_country].SiblingsCount()) + return root[index.m_group][index.m_country]; + return root[index.m_group][index.m_country][index.m_region]; } - return 0; // not found } - size_t Storage::GroupsCount() const + Country const & Storage::CountryByIndex(TIndex const & index) const { - return m_countries.size(); + return NodeFromIndex(m_countries, index).Value(); } - size_t Storage::CountriesCountInGroup(size_t groupIndex) const + size_t Storage::CountriesCount(TIndex const & index) const { - for (TCountriesContainer::const_iterator it = m_countries.begin(); it != m_countries.end(); ++it) - { - if (groupIndex == 0) - return it->second.size(); - else - --groupIndex; - } - return 0; - } - - string Storage::GroupName(size_t groupIndex) const - { - for (TCountriesContainer::const_iterator it = m_countries.begin(); it != m_countries.end(); ++it) - { - if (groupIndex == 0) - return it->first; - else - --groupIndex; - } - return string("Atlantida"); + return NodeFromIndex(m_countries, index).SiblingsCount(); } string Storage::CountryName(TIndex const & index) const { - Country const * country = CountryByIndex(index); - if (country) - return country->Name(); - else - { - ASSERT(false, ("Invalid country index", index)); - } - return string("Golden Land"); + return NodeFromIndex(m_countries, index).Value().Name(); } - uint64_t Storage::CountrySizeInBytes(TIndex const & index) const + TLocalAndRemoteSize Storage::CountrySizeInBytes(TIndex const & index) const { - Country const * country = CountryByIndex(index); - if (country) - { - uint64_t remoteSize = country->RemoteSize(); - return remoteSize ? remoteSize : country->LocalSize(); - } - else - { - ASSERT(false, ("Invalid country index", index)); - } - return 0; + return CountryByIndex(index).Size(); } TStatus Storage::CountryStatus(TIndex const & index) const @@ -193,16 +84,10 @@ namespace storage return EInQueue; } - Country const * country = CountryByIndex(index); - if (country) - { - if (country->RemoteSize() == 0) - return EOnDisk; - } - else - { - ASSERT(false, ("Invalid country index", index)); - } + TLocalAndRemoteSize size = CountryByIndex(index).Size(); + if (size.first == size.second) + return EOnDisk; + return ENotDownloaded; } @@ -219,11 +104,9 @@ namespace storage // and start download if necessary if (m_queue.size() == 1) { - Country const * country = CountryByIndex(index); - if (country) - { // reset total country's download progress - m_countryProgress = TDownloadProgress(0, country->RemoteSize()); - } + // reset total country's download progress + TLocalAndRemoteSize size = CountryByIndex(index).Size(); + m_countryProgress = TDownloadProgress(0, size.second); DownloadNextCountryFromQueue(); } @@ -244,11 +127,10 @@ namespace storage { m_workingDir = GetPlatform().WritableDir(); } - void operator()(TUrl const & url) + void operator()(TTile const & tile) { - string const file = m_workingDir + FileNameFromUrl(url.first); - if (IsDatFile(file)) - m_removeFn(file); + string const file = m_workingDir + tile.first; + m_removeFn(file); } }; @@ -257,31 +139,28 @@ namespace storage while (!m_queue.empty()) { TIndex index = m_queue.front(); - Country const * country = CountryByIndex(index); - if (country) + TTilesContainer const & tiles = CountryByIndex(index).Tiles(); + for (TTilesContainer::const_iterator it = tiles.begin(); it != tiles.end(); ++it) { - for (TUrlContainer::const_iterator it = country->Urls().begin(); it != country->Urls().end(); ++it) + if (!IsTileDownloaded(*it)) { - if (!IsFileDownloaded(*it)) - { - GetDownloadManager().DownloadFile( - it->first.c_str(), - (GetPlatform().WritablePathForFile(FileNameFromUrl(it->first))).c_str(), - boost::bind(&Storage::OnMapDownloadFinished, this, _1, _2), - boost::bind(&Storage::OnMapDownloadProgress, this, _1, _2), - true); // enabled resume support by default - // notify GUI - new status for country, "Downloading" - if (m_observerChange) - m_observerChange(index); - return; - } + GetDownloadManager().DownloadFile( + (UPDATE_BASE_URL + it->first).c_str(), + (GetPlatform().WritablePathForFile(it->first).c_str()), + boost::bind(&Storage::OnMapDownloadFinished, this, _1, _2), + boost::bind(&Storage::OnMapDownloadProgress, this, _1, _2), + true); // enabled resume support by default + // notify GUI - new status for country, "Downloading" + if (m_observerChange) + m_observerChange(index); + return; } } // continue with next country m_queue.pop_front(); // reset total country's download progress - if (!m_queue.empty() && (country = CountryByIndex(m_queue.front()))) - m_countryProgress = TDownloadProgress(0, country->RemoteSize()); + if (!m_queue.empty()) + m_countryProgress = TDownloadProgress(0, CountryByIndex(m_queue.front()).Size().second); // and notify GUI - new status for country, "OnDisk" if (m_observerChange) m_observerChange(index); @@ -290,16 +169,15 @@ namespace storage struct CancelDownloading { - void operator()(TUrl const & url) + void operator()(TTile const & tile) { - GetDownloadManager().CancelDownload(url.first.c_str()); - FileWriter::DeleteFile(GetPlatform().WritablePathForFile(FileNameFromUrl(url.first))); + GetDownloadManager().CancelDownload((UPDATE_BASE_URL + tile.first).c_str()); } }; void CancelCountryDownload(Country const & country) { - for_each(country.Urls().begin(), country.Urls().end(), CancelDownloading()); + for_each(country.Tiles().begin(), country.Tiles().end(), CancelDownloading()); } class DeleteMap @@ -310,10 +188,10 @@ namespace storage { m_workingDir = GetPlatform().WritableDir(); } - void operator()(TUrl const & url) + /// @TODO do not delete other countries cells + void operator()(TTile const & tile) { - string const file = m_workingDir + FileNameFromUrl(url.first); - FileWriter::DeleteFile(file); + FileWriter::DeleteFile(m_workingDir + tile.first); } }; @@ -321,26 +199,22 @@ namespace storage void DeactivateAndDeleteCountry(Country const & country, TRemoveFunc removeFunc) { // deactivate from multiindex - for_each(country.Urls().begin(), country.Urls().end(), DeactivateMap(removeFunc)); + for_each(country.Tiles().begin(), country.Tiles().end(), DeactivateMap(removeFunc)); // delete from disk - for_each(country.Urls().begin(), country.Urls().end(), DeleteMap()); + for_each(country.Tiles().begin(), country.Tiles().end(), DeleteMap()); } void Storage::DeleteCountry(TIndex const & index) { - Country const * country = CountryByIndex(index); - if (!country) - { - ASSERT(false, ("Invalid country index")); - return; - } + Country const & country = CountryByIndex(index); + // check if we already downloading this country TQueue::iterator found = find(m_queue.begin(), m_queue.end(), index); if (found != m_queue.end()) { if (found == m_queue.begin()) { // stop download - CancelCountryDownload(*country); + CancelCountryDownload(country); // remove from the queue m_queue.erase(found); // start another download if the queue is not empty @@ -353,7 +227,7 @@ namespace storage } // @TODO: Do not delete pieces which are used by other countries - DeactivateAndDeleteCountry(*country, m_removeMap); + DeactivateAndDeleteCountry(country, m_removeMap); if (m_observerChange) m_observerChange(index); } @@ -362,35 +236,39 @@ namespace storage { m_observerChange = change; m_observerProgress = progress; + + TTilesContainer tiles; + if (LoadTiles(tiles, GetPlatform().WritablePathForFile(UPDATE_CHECK_FILE))) + { + if (!LoadCountries(GetPlatform().WritablePathForFile(COUNTRIES_FILE), tiles, m_countries)) + LOG(LWARNING, ("Can't load countries file", COUNTRIES_FILE)); + } + else + { + LOG(LWARNING, ("Can't load update file", UPDATE_CHECK_FILE)); + } } void Storage::Unsubscribe() { m_observerChange.clear(); m_observerProgress.clear(); + m_countries.Clear(); } - bool IsUrlValidForCountry(char const * url, Country const * country) + string FileFromUrl(string const & url) { - // additional debug checking - if (country) - { - for (TUrlContainer::const_iterator it = country->Urls().begin(); it != country->Urls().end(); ++it) - if (it->first == url) - return true; - } - return false; + return url.substr(url.find_last_of('/') + 1, string::npos); } void Storage::OnMapDownloadFinished(char const * url, bool successfully) { - Country const * country = 0; - if (m_queue.empty() || !(country = CountryByIndex(m_queue.front()))) + if (m_queue.empty()) { ASSERT(false, ("Invalid url?", url)); return; } - ASSERT(IsUrlValidForCountry(url, country), ()); + if (!successfully) { // remove failed country from the queue @@ -402,17 +280,12 @@ namespace storage } else { - uint64_t rSize = country->RemoteSize(); - if (rSize == 0) - { // country is fully downloaded - // activate it! - // @TODO: activate only downloaded map, not all maps - RescanAndAddDownloadedMaps(m_addMap); - } - else - { - m_countryProgress.first = (m_countryProgress.second - rSize); - } + TLocalAndRemoteSize size = CountryByIndex(m_queue.front()).Size(); + if (size.second != 0) + m_countryProgress.first = (m_countryProgress.second - size.second); + // activate downloaded map piece + string const datFile = FileFromUrl(url); + m_addMap(datFile, datFile + INDEX_FILE_EXTENSION); } DownloadNextCountryFromQueue(); } @@ -439,23 +312,22 @@ namespace storage } // parse update file - TCountriesContainer tempCountries; - if (!LoadCountries(tempCountries, GetPlatform().WritablePathForFile(UPDATE_CHECK_FILE))) - { - LOG(LWARNING, ("New application version should be downloaded, " - "update file format can't be parsed")); - // @TODO: report to GUI - return; - } - // stop any active download, clear the queue, replace countries and notify GUI - if (!m_queue.empty()) - { - Country const * country = CountryByIndex(m_queue.front()); - CancelCountryDownload(*country); - m_queue.clear(); - } - m_countries.swap(tempCountries); - // @TODO report to GUI about reloading all countries - LOG(LINFO, ("Update check complete")); +// TCountriesContainer tempCountries; +// if (!LoadCountries(tempCountries, GetPlatform().WritablePathForFile(UPDATE_CHECK_FILE))) +// { +// LOG(LWARNING, ("New application version should be downloaded, " +// "update file format can't be parsed")); +// // @TODO: report to GUI +// return; +// } +// // stop any active download, clear the queue, replace countries and notify GUI +// if (!m_queue.empty()) +// { +// CancelCountryDownload(CountryByIndex(m_queue.front())); +// m_queue.clear(); +// } +// m_countries.swap(tempCountries); +// // @TODO report to GUI about reloading all countries +// LOG(LINFO, ("Update check complete")); } } diff --git a/storage/storage.hpp b/storage/storage.hpp index 39ce834dac..4dc9fa6baa 100644 --- a/storage/storage.hpp +++ b/storage/storage.hpp @@ -25,7 +25,18 @@ namespace storage EInQueue }; - typedef std::pair TIndex; + struct TIndex + { + int m_group; + int m_country; + int m_region; + TIndex(int group = -1, int country = -1, int region = -1) + : m_group(group), m_country(country), m_region(region) {} + bool operator==(TIndex const & other) const + { + return m_group == other.m_group && m_country == other.m_country && m_region == other.m_region; + } + }; /// Can be used to store local maps and/or maps available for download class Storage @@ -54,8 +65,8 @@ namespace storage //@} void DownloadNextCountryFromQueue(); - Country const * CountryByIndex(TIndex const & index) const; - bool UpdateCheck(); + Country const & CountryByIndex(TIndex const & index) const; +// bool UpdateCheck(); public: Storage() {} @@ -76,11 +87,9 @@ namespace storage void Unsubscribe(); //@} - size_t GroupsCount() const; - size_t CountriesCountInGroup(size_t groupIndex) const; - string GroupName(size_t groupIndex) const; + size_t CountriesCount(TIndex const & index) const; string CountryName(TIndex const & index) const; - uint64_t CountrySizeInBytes(TIndex const & index) const; + TLocalAndRemoteSize CountrySizeInBytes(TIndex const & index) const; TStatus CountryStatus(TIndex const & index) const; void DownloadCountry(TIndex const & index); diff --git a/storage/storage.pro b/storage/storage.pro index 28ec06fe77..4ebce898c5 100644 --- a/storage/storage.pro +++ b/storage/storage.pro @@ -5,16 +5,16 @@ TEMPLATE = lib CONFIG += staticlib ROOT_DIR = .. -DEPENDENCIES = geometry coding base +DEPENDENCIES = indexer platform geometry coding base include($$ROOT_DIR/common.pri) +HEADERS += \ + country.hpp \ + defines.hpp \ + simple_tree.hpp \ + storage.hpp \ + SOURCES += \ country.cpp \ storage.cpp \ - -HEADERS += \ - defines.hpp \ - country.hpp \ - simple_tree.hpp \ - storage.hpp \ diff --git a/storage/storage_tests/country_test.cpp b/storage/storage_tests/country_test.cpp index 181e218c8b..5c6898575d 100644 --- a/storage/storage_tests/country_test.cpp +++ b/storage/storage_tests/country_test.cpp @@ -11,32 +11,32 @@ using namespace storage; UNIT_TEST(CountrySerialization) { - string const TEST_URL = "http://someurl.com/somemap.dat"; - uint64_t const TEST_SIZE = 123456790; - char const * TEST_FILE_NAME = "some_temporary_update_file.tmp"; - Country c("North America", "USA", "Alaska"); - c.AddUrl(TUrl(TEST_URL, TEST_SIZE)); +// string const TEST_URL = "http://someurl.com/somemap.dat"; +// uint64_t const TEST_SIZE = 123456790; +// char const * TEST_FILE_NAME = "some_temporary_update_file.tmp"; +// Country c("North America", "USA", "Alaska"); +// c.AddUrl(TUrl(TEST_URL, TEST_SIZE)); - { - TCountriesContainer countries; - countries[c.Group()].push_back(c); +// { +// TCountriesContainer countries; +// countries[c.Group()].push_back(c); - FileWriter writer(TEST_FILE_NAME); - SaveCountries(countries, writer); - } +// FileWriter writer(TEST_FILE_NAME); +// SaveCountries(countries, writer); +// } - TCountriesContainer loadedCountries; - { - TEST( LoadCountries(loadedCountries, TEST_FILE_NAME), ()); - } +// TCountriesContainer loadedCountries; +// { +// TEST( LoadCountries(loadedCountries, TEST_FILE_NAME), ()); +// } - TEST_GREATER(loadedCountries.size(), 0, ()); - Country const & c2 = loadedCountries.begin()->second.front(); - TEST_EQUAL(c.Group(), loadedCountries.begin()->first, ()); - TEST_EQUAL(c.Group(), c2.Group(), ()); - TEST_EQUAL(c.Name(), c2.Name(), ()); - TEST_GREATER(c2.Urls().size(), 0, ()); - TEST_EQUAL(*c.Urls().begin(), *c2.Urls().begin(), ()); +// TEST_GREATER(loadedCountries.size(), 0, ()); +// Country const & c2 = loadedCountries.begin()->second.front(); +// TEST_EQUAL(c.Group(), loadedCountries.begin()->first, ()); +// TEST_EQUAL(c.Group(), c2.Group(), ()); +// TEST_EQUAL(c.Name(), c2.Name(), ()); +// TEST_GREATER(c2.Urls().size(), 0, ()); +// TEST_EQUAL(*c.Urls().begin(), *c2.Urls().begin(), ()); - FileWriter::DeleteFile(TEST_FILE_NAME); +// FileWriter::DeleteFile(TEST_FILE_NAME); } diff --git a/storage/storage_tests/simple_tree_test.cpp b/storage/storage_tests/simple_tree_test.cpp index 4d254c04f5..28b892249d 100644 --- a/storage/storage_tests/simple_tree_test.cpp +++ b/storage/storage_tests/simple_tree_test.cpp @@ -1,13 +1,10 @@ #include "../../testing/testing.hpp" -#include "../../platform/platform.hpp" - -#include "../../std/fstream.hpp" +#include "../../coding/reader.hpp" +#include "../../coding/writer.hpp" +#include "../../coding/streams_sink.hpp" #include "../simple_tree.hpp" -#include "../countries.hpp" - -using namespace mapinfo; template struct Calculator @@ -22,34 +19,39 @@ struct Calculator UNIT_TEST(SimpleTree) { - typedef SimpleTree mytree; + typedef SimpleTree mytree; mytree tree; - std::ifstream file(GetPlatform().WritablePathForFile("countries.txt").c_str()); - TEST( LoadCountries(tree, file), ("countries.txt is absent or corrupted")); + + tree.Add(4); + tree.Add(3); + tree.Add(5); + tree.Add(2); + tree.Add(1); + tree.AddAtDepth(1, 20); // 1 is parent + tree.AddAtDepth(1, 10); // 1 is parent + tree.AddAtDepth(1, 30); // 1 is parent tree.Sort(); + // test sorting + TEST_EQUAL(tree[0].Value(), 1, ()); + TEST_EQUAL(tree[1].Value(), 2, ()); + TEST_EQUAL(tree[2].Value(), 3, ()); + TEST_EQUAL(tree[3].Value(), 4, ()); + TEST_EQUAL(tree[4].Value(), 5, ()); + TEST_EQUAL(tree[0][0].Value(), 10, ()); + TEST_EQUAL(tree[0][1].Value(), 20, ()); + TEST_EQUAL(tree[0][2].Value(), 30, ()); Calculator c1; tree.ForEachSibling(c1); - TEST_GREATER_OR_EQUAL(c1.count, 6, ("At least 6 continents should be present")); - - // find europe - size_t i = 0; - for (; i < tree.SiblingsCount(); ++i) - if (tree[i].Value().Name() == "Europe") - break; - TEST_LESS(i, tree.SiblingsCount(), ("Europe is not present?")); + TEST_EQUAL(c1.count, 5, ()); Calculator c2; - tree[i].ForEachSibling(c2); - TEST_GREATER_OR_EQUAL(c2.count, 20, ("At least 20 countries should be present in Europe")); - - Calculator c3; - tree[i].ForEachChildren(c3); - TEST_GREATER(c3.count, c2.count, ("Europe should contain some regions")); + tree.ForEachChildren(c2); + TEST_EQUAL(c2.count, 8, ()); tree.Clear(); - Calculator c4; - tree.ForEachChildren(c4); - TEST_EQUAL(c4.count, 0, ("Tree should be empty")); + Calculator c3; + tree.ForEachChildren(c3); + TEST_EQUAL(c3.count, 0, ("Tree should be empty")); } diff --git a/storage/storage_tests/storage_tests.pro b/storage/storage_tests/storage_tests.pro index 4dd164ee52..0297932920 100644 --- a/storage/storage_tests/storage_tests.pro +++ b/storage/storage_tests/storage_tests.pro @@ -6,7 +6,7 @@ CONFIG -= app_bundle TEMPLATE = app ROOT_DIR = ../.. -DEPENDENCIES = storage geometry platform coding base +DEPENDENCIES = storage indexer geometry platform coding base include($$ROOT_DIR/common.pri) @@ -16,4 +16,5 @@ HEADERS += SOURCES += \ ../../testing/testingmain.cpp \ -# simple_tree_test.cpp \ + country_test.cpp \ + simple_tree_test.cpp \