From 02d98d9e0ce2704e717032ce4fbd2b5df6b58c85 Mon Sep 17 00:00:00 2001 From: Vladimir Byko-Ianko Date: Tue, 1 Nov 2016 16:38:33 +0300 Subject: [PATCH] Using different input files for restrictions and feature id to osm ids mapping to form restrictions in feature id terms. --- generator/generate_info.hpp | 5 + generator/generator.pro | 3 + .../generator_tests/restrictions_tests.cpp | 142 ++++++++++++--- generator/generator_tool/generator_tool.cpp | 17 +- generator/osm_source.cpp | 8 +- generator/restriction_dumper.cpp | 2 +- generator/restriction_generator.cpp | 22 +++ generator/restriction_generator.hpp | 9 + generator/restrictions.cpp | 166 +++++++++++++----- generator/restrictions.hpp | 86 +++++---- 10 files changed, 350 insertions(+), 110 deletions(-) create mode 100644 generator/restriction_generator.cpp create mode 100644 generator/restriction_generator.hpp diff --git a/generator/generate_info.hpp b/generator/generate_info.hpp index 275ac2e9d8..9733060fee 100644 --- a/generator/generate_info.hpp +++ b/generator/generate_info.hpp @@ -36,6 +36,11 @@ struct GenerateInfo // Current generated file name if --output option is defined. string m_fileName; + // File name with restriction in osm id terms. + string m_restrictions; + // File name with mapping from feature id to osm ids. + string m_featureId2OsmIds; + NodeStorageType m_nodeStorageType; OsmSourceType m_osmFileType; string m_osmFileName; diff --git a/generator/generator.pro b/generator/generator.pro index 145c6aea36..40fa8a0675 100644 --- a/generator/generator.pro +++ b/generator/generator.pro @@ -37,6 +37,7 @@ SOURCES += \ osm_source.cpp \ region_meta.cpp \ restriction_dumper.cpp \ + restriction_generator.cpp \ restrictions.cpp \ routing_generator.cpp \ search_index_builder.cpp \ @@ -76,6 +77,7 @@ HEADERS += \ polygonizer.hpp \ region_meta.hpp \ restriction_dumper.hpp \ + restriction_generator.hpp \ restrictions.hpp \ routing_generator.hpp \ search_index_builder.hpp \ @@ -90,3 +92,4 @@ HEADERS += \ unpack_mwm.hpp \ ways_merger.hpp \ world_map_generator.hpp \ + diff --git a/generator/generator_tests/restrictions_tests.cpp b/generator/generator_tests/restrictions_tests.cpp index 604c6b6385..6ab48027c7 100644 --- a/generator/generator_tests/restrictions_tests.cpp +++ b/generator/generator_tests/restrictions_tests.cpp @@ -3,31 +3,40 @@ #include "generator/osm_id.hpp" #include "generator/restrictions.hpp" +#include "coding/file_name_utils.hpp" + +#include "platform/platform_tests_support/scoped_dir.hpp" +#include "platform/platform_tests_support/scoped_file.hpp" + +#include "platform/platform.hpp" + +#include "std/string.hpp" #include "std/vector.hpp" +using namespace platform; +using namespace platform::tests_support; + +string const kRestrictionTestDir = "test-restrictions"; + UNIT_TEST(RestrictionTest_ValidCase) { - RestrictionCollector restrictionCollector; - + RestrictionCollector restrictionCollector("", ""); // Adding restrictions and feature ids to restrictionCollector in mixed order. - restrictionCollector.AddRestriction({osm::Id::Way(1), osm::Id::Way(2)}, - RestrictionCollector::Type::No); - restrictionCollector.AddFeatureId({osm::Id::Way(3)}, 30 /* featureId */); - restrictionCollector.AddRestriction({osm::Id::Way(2), osm::Id::Way(3)}, - RestrictionCollector::Type::No); - restrictionCollector.AddFeatureId({osm::Id::Way(1)}, 10 /* featureId */); - restrictionCollector.AddFeatureId({osm::Id::Way(5)}, 50 /* featureId */); - restrictionCollector.AddRestriction({osm::Id::Way(5), osm::Id::Way(7)}, - RestrictionCollector::Type::Only); - restrictionCollector.AddFeatureId({osm::Id::Way(7)}, 70 /* featureId */); - restrictionCollector.AddFeatureId({osm::Id::Way(2)}, 20 /* featureId */); + restrictionCollector.AddRestriction(RestrictionCollector::Type::No, {1, 2} /* osmIds */); + restrictionCollector.AddFeatureId(30 /* featureId */, {3} /* osmIds */); + restrictionCollector.AddRestriction(RestrictionCollector::Type::No, {2, 3} /* osmIds */); + restrictionCollector.AddFeatureId(10 /* featureId */, {1} /* osmIds */); + restrictionCollector.AddFeatureId(50 /* featureId */, {5} /* osmIds */); + restrictionCollector.AddRestriction(RestrictionCollector::Type::Only, {5, 7} /* osmIds */); + restrictionCollector.AddFeatureId(70 /* featureId */, {7} /* osmIds */); + restrictionCollector.AddFeatureId(20 /* featureId */, {2} /* osmIds */); // Composing restriction in feature id terms. restrictionCollector.ComposeRestrictions(); restrictionCollector.RemoveInvalidRestrictions(); // Checking the result. - TEST(restrictionCollector.CheckCorrectness(), ()); + TEST(restrictionCollector.IsValid(), ()); vector const expectedRestrictions = {{RestrictionCollector::Type::No, {10, 20}}, @@ -38,15 +47,14 @@ UNIT_TEST(RestrictionTest_ValidCase) UNIT_TEST(RestrictionTest_InvalidCase) { - RestrictionCollector restrictionCollector; - restrictionCollector.AddFeatureId({osm::Id::Way(0)}, 0 /* featureId */); - restrictionCollector.AddRestriction({osm::Id::Way(0), osm::Id::Way(1)}, - RestrictionCollector::Type::No); - restrictionCollector.AddFeatureId({osm::Id::Way(2)}, 20 /* featureId */); + RestrictionCollector restrictionCollector("", ""); + restrictionCollector.AddFeatureId(0 /* featureId */, {0} /* osmIds */); + restrictionCollector.AddRestriction(RestrictionCollector::Type::No, {0, 1} /* osmIds */); + restrictionCollector.AddFeatureId(20 /* featureId */, {2} /* osmIds */); restrictionCollector.ComposeRestrictions(); - TEST(!restrictionCollector.CheckCorrectness(), ()); + TEST(!restrictionCollector.IsValid(), ()); vector const expectedRestrictions = {{RestrictionCollector::Type::No, {0, RestrictionCollector::kInvalidFeatureId}}}; @@ -54,5 +62,97 @@ UNIT_TEST(RestrictionTest_InvalidCase) restrictionCollector.RemoveInvalidRestrictions(); TEST(restrictionCollector.m_restrictions.empty(), ()); - TEST(restrictionCollector.CheckCorrectness(), ()); + TEST(!restrictionCollector.IsValid(), ()); +} + +UNIT_TEST(RestrictionTest_ParseRestrictions) +{ + string const kRestrictionName = "restrictions_in_osm_ids.csv"; + string const kRestrictionPath = my::JoinFoldersToPath(kRestrictionTestDir, kRestrictionName); + string const kRestrictionContent = R"(No, 1, 1, + Only, 0, 2, + Only, 2, 3, + No, 38028428, 38028428, + No, 4, 5,)"; + + ScopedDir const scopedDir(kRestrictionTestDir); + ScopedFile const scopedFile(kRestrictionPath, kRestrictionContent); + + RestrictionCollector restrictionCollector("", ""); + + Platform const & platform = Platform(); + + TEST(restrictionCollector.ParseRestrictions(my::JoinFoldersToPath(platform.WritableDir(), + kRestrictionPath)), ()); + vector expectedRestrictions = + {{RestrictionCollector::Type::No, 2}, + {RestrictionCollector::Type::Only, 2}, + {RestrictionCollector::Type::Only, 2}, + {RestrictionCollector::Type::No, 2}, + {RestrictionCollector::Type::No, 2}}; + TEST_EQUAL(restrictionCollector.m_restrictions, expectedRestrictions, ()); + + vector> const expectedRestrictionIndex = + {{1, {0, 0}}, {1, {0, 1}}, + {0, {1, 0}}, {2, {1, 1}}, + {2, {2, 0}}, {3, {2, 1}}, + {38028428, {3, 0}}, {38028428, {3, 1}}, + {4, {4, 0}}, {5, {4, 1}}}; + TEST_EQUAL(restrictionCollector.m_restrictionIndex, expectedRestrictionIndex, ()); +} + +UNIT_TEST(RestrictionTest_ParseFeatureId2OsmIdsMapping) +{ + string const kFeatureIdToOsmIdsName = "feature_id_to_osm_ids.csv"; + string const kFeatureIdToOsmIdsPath = my::JoinFoldersToPath(kRestrictionTestDir, kFeatureIdToOsmIdsName); + string const kFeatureIdToOsmIdsContent = R"(1, 10, + 2, 20, + 779703, 5423239545, + 3, 30)"; + + ScopedDir const scopedDir(kRestrictionTestDir); + ScopedFile const scopedFile(kFeatureIdToOsmIdsPath, kFeatureIdToOsmIdsContent); + + RestrictionCollector restrictionCollector("", ""); + + Platform const & platform = Platform(); + restrictionCollector.ParseFeatureId2OsmIdsMapping(my::JoinFoldersToPath(platform.WritableDir(), + kFeatureIdToOsmIdsPath)); + + vector> const expectedOsmIds2FeatureId = + {{10, 1}, {20, 2}, {5423239545, 779703}, {30, 3}}; + vector> const osmIds2FeatureId( + restrictionCollector.m_osmIds2FeatureId.cbegin(), restrictionCollector.m_osmIds2FeatureId.cend()); + TEST_EQUAL(osmIds2FeatureId, expectedOsmIds2FeatureId, ()); +} + +UNIT_TEST(RestrictionTest_RestrictionCollectorWholeClassTest) +{ + string const kRestrictionName = "restrictions_in_osm_ids.csv"; + string const kRestrictionPath = my::JoinFoldersToPath(kRestrictionTestDir, kRestrictionName); + string const kRestrictionContent = R"(No, 10, 10, + Only, 10, 20, + Only, 30, 40,)"; + + string const kFeatureIdToOsmIdsName = "feature_id_to_osm_ids.csv"; + string const kFeatureIdToOsmIdsPath = my::JoinFoldersToPath(kRestrictionTestDir, kFeatureIdToOsmIdsName); + string const kFeatureIdToOsmIdsContent = R"(1, 10, + 2, 20, + 3, 30, + 4, 40)"; + + ScopedDir scopedDir(kRestrictionTestDir); + ScopedFile restrictionScopedFile(kRestrictionPath, kRestrictionContent); + ScopedFile mappingScopedFile(kFeatureIdToOsmIdsPath, kFeatureIdToOsmIdsContent); + + Platform const & platform = Platform(); + RestrictionCollector restrictionCollector(my::JoinFoldersToPath(platform.WritableDir(), kRestrictionPath), + my::JoinFoldersToPath(platform.WritableDir(), kFeatureIdToOsmIdsPath)); + TEST(restrictionCollector.IsValid(), ()); + + vector const expectedRestrictions = + {{RestrictionCollector::Type::No, {1, 1}}, + {RestrictionCollector::Type::Only, {1, 2}}, + {RestrictionCollector::Type::Only, {3, 4}}}; + TEST_EQUAL(restrictionCollector.GetRestriction(), expectedRestrictions, ()); } diff --git a/generator/generator_tool/generator_tool.cpp b/generator/generator_tool/generator_tool.cpp index 6024c0d679..c4c9e3687b 100644 --- a/generator/generator_tool/generator_tool.cpp +++ b/generator/generator_tool/generator_tool.cpp @@ -8,6 +8,7 @@ #include "generator/feature_sorter.hpp" #include "generator/generate_info.hpp" #include "generator/osm_source.hpp" +#include "generator/restriction_generator.hpp" #include "generator/routing_generator.hpp" #include "generator/search_index_builder.hpp" #include "generator/statistics.hpp" @@ -82,6 +83,9 @@ DEFINE_uint64(planet_version, my::SecondsSinceEpoch(), DEFINE_string(srtm_path, "", "Path to srtm directory. If it is set, generates section with altitude information " "about roads."); +DEFINE_string(restriction_name, "", "Name of file with relation restriction in osm id term."); +DEFINE_string(feature_id_to_osm_ids_name, "", "Name of to file with mapping from feature id to osm ids."); +DEFINE_bool(generate_outgoing_edge_index, false, "Generates section with outgoing edges"); int main(int argc, char ** argv) { @@ -112,6 +116,8 @@ int main(int argc, char ** argv) } genInfo.m_osmFileName = FLAGS_osm_file_name; + genInfo.m_restrictions = FLAGS_restriction_name; + genInfo.m_featureId2OsmIds = FLAGS_feature_id_to_osm_ids_name; genInfo.m_failOnCoasts = FLAGS_fail_on_coasts; genInfo.m_preloadCache = FLAGS_preload_cache; genInfo.m_bookingDatafileName = FLAGS_booking_data; @@ -143,7 +149,8 @@ int main(int argc, char ** argv) if (FLAGS_make_coasts || FLAGS_generate_features || FLAGS_generate_geometry || FLAGS_generate_index || FLAGS_generate_search_index || FLAGS_calc_statistics || FLAGS_type_statistics || FLAGS_dump_types || FLAGS_dump_prefixes || - FLAGS_dump_feature_names != "" || FLAGS_check_mwm || FLAGS_srtm_path != "") + FLAGS_dump_feature_names != "" || FLAGS_check_mwm || FLAGS_srtm_path != "" || + FLAGS_generate_outgoing_edge_index) { classificator::Load(); classif().SortClassificator(); @@ -154,6 +161,8 @@ int main(int argc, char ** argv) { LOG(LINFO, ("Generating final data ...")); + genInfo.m_restrictions = FLAGS_restriction_name; + genInfo.m_featureId2OsmIds = FLAGS_feature_id_to_osm_ids_name; genInfo.m_splitByPolygons = FLAGS_split_by_polygons; genInfo.m_createWorld = FLAGS_generate_world; genInfo.m_makeCoasts = FLAGS_make_coasts; @@ -228,6 +237,12 @@ int main(int argc, char ** argv) if (!FLAGS_srtm_path.empty()) routing::BuildRoadAltitudes(datFile, FLAGS_srtm_path); + + if (FLAGS_generate_outgoing_edge_index) + { + routing::BuildRoadRestrictions(datFile, genInfo.GetIntermediateFileName(genInfo.m_restrictions, ""), + genInfo.GetIntermediateFileName(genInfo.m_featureId2OsmIds, "")); + } } string const datFile = my::JoinFoldersToPath(path, FLAGS_output + DATA_FILE_EXTENSION); diff --git a/generator/osm_source.cpp b/generator/osm_source.cpp index 1aeddefe7c..30ad13d308 100644 --- a/generator/osm_source.cpp +++ b/generator/osm_source.cpp @@ -343,7 +343,7 @@ public: m_world.reset(new TWorldGenerator(info)); // Feature id osm id to map. - string const featureId2OsmIdsFile = info.GetIntermediateFileName("feature_id_to_osm_ids", ".csv"); + string const featureId2OsmIdsFile = info.GetIntermediateFileName(info.m_featureId2OsmIds, ""); LOG(LINFO, ("Saving osm ids to feature ids map to", featureId2OsmIdsFile)); m_featureId2osmIds.Open(featureId2OsmIdsFile); if (!m_featureId2osmIds.IsOpened()) @@ -550,9 +550,9 @@ void SyncOfstream::Write(uint32_t featureId, vector const & osmIds) return; lock_guard gard(m_mutex); - m_stream << featureId << ", "; + m_stream << featureId << ","; for (osm::Id const & osmId : osmIds) - m_stream << osmId.OsmId() << ", "; + m_stream << osmId.OsmId() << ","; m_stream << endl; } @@ -720,7 +720,7 @@ bool GenerateFeaturesImpl(feature::GenerateInfo & info, EmitterBase & emitter) // TODO(mgsergio): Get rid of EmitterBase template parameter. OsmToFeatureTranslator parser( emitter, cache, info.m_makeCoasts ? classif().GetCoastType() : 0, - info.GetAddressesFileName(), info.GetIntermediateFileName("restrictions_in_osm_ids", ".csv")); + info.GetAddressesFileName(), info.GetIntermediateFileName(info.m_restrictions, "")); TagAdmixer tagAdmixer(info.GetIntermediateFileName("ways", ".csv"), info.GetIntermediateFileName("towns", ".csv")); diff --git a/generator/restriction_dumper.cpp b/generator/restriction_dumper.cpp index 46cf7ced8e..3068c5739c 100644 --- a/generator/restriction_dumper.cpp +++ b/generator/restriction_dumper.cpp @@ -86,7 +86,7 @@ void RestrictionDumper::Write(RelationElement const & relationElement) return; // Unsupported restriction type. // Adding restriction. - m_stream << ToString(typeResult.first) << ", " // Restriction type + m_stream << ToString(typeResult.first) << "," // Restriction type << fromIt->first << ", " << toIt->first << "," << endl; } diff --git a/generator/restriction_generator.cpp b/generator/restriction_generator.cpp new file mode 100644 index 0000000000..bad3cef9e8 --- /dev/null +++ b/generator/restriction_generator.cpp @@ -0,0 +1,22 @@ +#include "generator/restriction_generator.hpp" +#include "generator/restrictions.hpp" + +#include "base/logging.hpp" + +namespace routing +{ +bool BuildRoadRestrictions(string const & mwmPath, string const & restrictionPath, + string const & featureId2OsmIdsPath) +{ + LOG(LINFO, ("BuildRoadRestrictions(", mwmPath, ", ", restrictionPath, ", ", featureId2OsmIdsPath, ");")); + RestrictionCollector restrictionCollector(restrictionPath, featureId2OsmIdsPath); + if (!restrictionCollector.IsValid()) + { + LOG(LWARNING, ("No valid restrictions for", mwmPath, "It's necessary to check if", + restrictionPath, "and", featureId2OsmIdsPath, "are available.")); + return false; + } + + return true; +} +} // namespace routing diff --git a/generator/restriction_generator.hpp b/generator/restriction_generator.hpp new file mode 100644 index 0000000000..a170a0b8ad --- /dev/null +++ b/generator/restriction_generator.hpp @@ -0,0 +1,9 @@ +#pragma once + +#include "std/string.hpp" + +namespace routing +{ + bool BuildRoadRestrictions(string const & mwmPath, string const & restrictionPath, + string const & featureId2OsmIdsPath); +} // namespace routing diff --git a/generator/restrictions.cpp b/generator/restrictions.cpp index 6af8fcf737..fd79c41779 100644 --- a/generator/restrictions.cpp +++ b/generator/restrictions.cpp @@ -3,16 +3,36 @@ #include "base/assert.hpp" #include "base/logging.hpp" #include "base/stl_helpers.hpp" +#include "base/string_utils.hpp" #include "std/algorithm.hpp" #include "std/fstream.hpp" -#include "std/string.hpp" -#include "std/vector.hpp" +#include "std/sstream.hpp" namespace { string const kNoStr = "No"; string const kOnlyStr = "Only"; + +bool ParseLineOfNumbers(istringstream & stream, vector & numbers) +{ + string numberStr; + uint64_t number; + + while (stream) + { + if (!getline(stream, numberStr, ',' )) + return true; + if (numberStr.empty()) + return true; + + if (!strings::to_uint64(numberStr, number)) + return false; + + numbers.push_back(number); + } + return true; +} } // namespace RestrictionCollector::FeatureId const RestrictionCollector::kInvalidFeatureId = @@ -46,50 +66,95 @@ bool RestrictionCollector::Restriction::operator<(Restriction const & restrictio return m_links < restriction.m_links; } -void RestrictionCollector::AddRestriction(vector const & links, Type type) +RestrictionCollector::RestrictionCollector(string const & restrictionPath, + string const & featureId2OsmIdsPath) { - lock_guard lock(m_mutex); - m_restrictions.emplace_back(type, links.size()); - size_t const restrictionCount = m_restrictions.size() - 1; - for (size_t i = 0; i < links.size(); ++i) - m_restrictionIndex.emplace_back(links[i], Index({restrictionCount, i})); + if (!ParseFeatureId2OsmIdsMapping(featureId2OsmIdsPath)) + return; + + if (!ParseRestrictions(restrictionPath)) + { + m_restrictions.clear(); + return; + } + ComposeRestrictions(); + RemoveInvalidRestrictions(); + LOG(LINFO, ("m_restrictions.size() =", m_restrictions.size())); } -void RestrictionCollector::AddFeatureId(vector const & osmIds, FeatureId featureId) +bool RestrictionCollector::IsValid() const { - // Note. One |featureId| could correspond to several osm ids. - // but for road feature |featureId| corresponds exactly one osm id. - lock_guard lock(m_mutex); - for (osm::Id const & osmId : osmIds) - m_osmIds2FeatureId.insert(make_pair(osmId, featureId)); + return !m_restrictions.empty() + && find_if(begin(m_restrictions), end(m_restrictions), + [](Restriction const & r){ return !r.IsValid(); }) == end(m_restrictions); } -bool RestrictionCollector::CheckCorrectness() const -{ - return find_if(begin(m_restrictions), end(m_restrictions), - [](Restriction const & r){ return !r.IsValid(); }) - == end(m_restrictions); +bool RestrictionCollector::ParseFeatureId2OsmIdsMapping(string const & featureId2OsmIdsPath) +{ + ifstream featureId2OsmIdsStream(featureId2OsmIdsPath); + if (featureId2OsmIdsStream.fail()) + return false; + + while (featureId2OsmIdsStream) + { + string line; + if (!getline(featureId2OsmIdsStream, line)) + return true; + + istringstream lineStream(line); + vector ids; + if (!ParseLineOfNumbers(lineStream, ids)) + return false; + + if (ids.size() <= 1) + return false; // Every line should contain at least feature id and osm id. + + FeatureId const featureId = static_cast(ids.front()); + ids.erase(ids.begin()); + AddFeatureId(featureId, ids); + } + return true; } -void RestrictionCollector::RemoveInvalidRestrictions() +bool RestrictionCollector::ParseRestrictions(string const & restrictionPath) { - m_restrictions.erase(remove_if(m_restrictions.begin(), m_restrictions.end(), - [](Restriction const & r){ return !r.IsValid(); }), - m_restrictions.end()); + ifstream restrictionsStream(restrictionPath); + if (restrictionsStream.fail()) + return false; + + while (restrictionsStream) + { + string line; + if (!getline(restrictionsStream, line)) + return true; + istringstream lineStream(line); + string typeStr; + getline(lineStream, typeStr, ',' ); + Type type; + if (!FromString(typeStr, type)) + return false; + + vector osmIds; + if (!ParseLineOfNumbers(lineStream, osmIds)) + return false; + + AddRestriction(type, osmIds); + } + return true; } void RestrictionCollector::ComposeRestrictions() { // Going throught all osm id saved in |m_restrictionIndex| (mentioned in restrictions). size_t const restrictionSz = m_restrictions.size(); - for (pair const & osmIdAndIndex : m_restrictionIndex) + for (pair const & osmIdAndIndex : m_restrictionIndex) { Index const & index = osmIdAndIndex.second; CHECK_LESS(index.m_restrictionNumber, restrictionSz, ()); Restriction & restriction = m_restrictions[index.m_restrictionNumber]; CHECK_LESS(index.m_linkNumber, restriction.m_links.size(), ()); - osm::Id const & osmId = osmIdAndIndex.first; + uint64_t const & osmId = osmIdAndIndex.first; // Checking if there's an osm id belongs to a restriction is saved only once as feature id. auto const rangeId = m_osmIds2FeatureId.equal_range(osmId); if (rangeId.first == rangeId.second) @@ -112,31 +177,27 @@ void RestrictionCollector::ComposeRestrictions() m_restrictionIndex.clear(); } -void RestrictionCollector::ComposeRestrictionsAndSave(string const & fullPath) +void RestrictionCollector::RemoveInvalidRestrictions() { - lock_guard lock(m_mutex); + m_restrictions.erase(remove_if(m_restrictions.begin(), m_restrictions.end(), + [](Restriction const & r){ return !r.IsValid(); }), + m_restrictions.end()); +} - ComposeRestrictions(); - RemoveInvalidRestrictions(); +void RestrictionCollector::AddRestriction(Type type, vector const & osmIds) +{ + m_restrictions.emplace_back(type, osmIds.size()); + size_t const restrictionCount = m_restrictions.size() - 1; + for (size_t i = 0; i < osmIds.size(); ++i) + m_restrictionIndex.emplace_back(osmIds[i], Index({restrictionCount, i})); +} - if (m_restrictions.empty()) - return; - - LOG(LINFO, ("Saving intermediate file with restrictions to", fullPath)); - ofstream ofs(fullPath, std::ofstream::out); - if (ofs.fail()) - { - LOG(LERROR, ("Cannot open", fullPath, "while saving road restrictions in intermediate format.")); - return; - } - - for (Restriction const & r : m_restrictions) - { - ofs << ToString(r.m_type) << ", "; - for (FeatureId fid : r.m_links) - ofs << fid << ", "; - ofs << endl; - } +void RestrictionCollector::AddFeatureId(FeatureId featureId, vector const & osmIds) +{ + // Note. One |featureId| could correspond to several osm ids. + // but for road feature |featureId| corresponds exactly one osm id. + for (uint64_t const & osmId : osmIds) + m_osmIds2FeatureId.insert(make_pair(osmId, featureId)); } string ToString(RestrictionCollector::Type const & type) @@ -151,8 +212,9 @@ string ToString(RestrictionCollector::Type const & type) return "Unknown"; } -bool FromString(string const & str, RestrictionCollector::Type & type) +bool FromString(string str, RestrictionCollector::Type & type) { + str.erase(remove_if(str.begin(), str.end(), isspace), str.end()); if (str == kNoStr) { type = RestrictionCollector::Type::No; @@ -172,6 +234,14 @@ string DebugPrint(RestrictionCollector::Type const & type) return ToString(type); } +string DebugPrint(RestrictionCollector::Index const & index) +{ + ostringstream out; + out << "m_restrictionNumber:" << index.m_restrictionNumber + << " m_linkNumber:" << index.m_linkNumber << " "; + return out.str(); +} + string DebugPrint(RestrictionCollector::Restriction const & restriction) { ostringstream out; diff --git a/generator/restrictions.hpp b/generator/restrictions.hpp index 4cd2b3bfb5..73dfa4f433 100644 --- a/generator/restrictions.hpp +++ b/generator/restrictions.hpp @@ -1,30 +1,22 @@ #pragma once -#include "generator/osm_id.hpp" - #include "std/functional.hpp" #include "std/limits.hpp" -#include "std/mutex.hpp" +#include "std/string.hpp" #include "std/unordered_map.hpp" #include "std/utility.hpp" #include "std/vector.hpp" class RelationElement; -template <> struct hash -{ - size_t operator()(osm::Id const & id) const - { - return hash()(id.OsmId()); - } -}; - /// This class collects all relations with type restriction and save feature ids of /// their road feature in text file for using later. class RestrictionCollector { friend void UnitTest_RestrictionTest_ValidCase(); friend void UnitTest_RestrictionTest_InvalidCase(); + friend void UnitTest_RestrictionTest_ParseRestrictions(); + friend void UnitTest_RestrictionTest_ParseFeatureId2OsmIdsMapping(); public: using FeatureId = uint32_t; static FeatureId const kInvalidFeatureId; @@ -63,49 +55,73 @@ public: { size_t m_restrictionNumber; // Restriction number in restriction vector. size_t m_linkNumber; // Link number for a restriction. It's equal to zero or one for most cases. + + bool operator==(Index const & index) const + { + return m_restrictionNumber == index.m_restrictionNumber + && m_linkNumber == index.m_linkNumber; + } }; - /// \brief Adds feature id and corresponding vector of |osmIds| to |m_osmId2FeatureId|. - /// \note One feature id (|featureId|) may correspond to several osm ids (|osmIds|). - void AddFeatureId(vector const & osmIds, FeatureId featureId); + /// \param restrictionPath full path to file with road restrictions in osm id terms. + /// \param featureId2OsmIdsPath full path to file with mapping from feature id to osm id. + RestrictionCollector(string const & restrictionPath, string const & featureId2OsmIdsPath); - /// \brief Save all restrictions (content of |m_restrictions|) to text file in the following format: - /// * One restriction is saved in one line - /// * Line format is: , , , ... - /// * Example 1: "No, 12345, 12346" - /// * Example 2: "Only, 12349, 12341" - /// \note Most restrictions consist of type and two linear(road) features. - /// \note For the time being only line-point-line restritions are supported. - void ComposeRestrictionsAndSave(string const & fullPath); + /// \returns true if |m_restrictions| is not empty all feature ids in |m_restrictions| + /// are set to valid value and false otherwise. + /// \note Empty |m_restrictions| is considered as an invalid restriction. + /// \note Complexity of the method is up to linear in the size of |m_restrictions|. + bool IsValid() const; + + vector const & GetRestriction() { return m_restrictions; } private: - /// \returns true if all feature ids in |m_restrictions| are set to valid value - /// and false otherwise. - /// \note The method may be called after ComposeRestrictions(). - bool CheckCorrectness() const; + /// \brief Parses comma separated text file with line in following format: + /// , , , and so on + /// For example: + /// 137999, 5170186, + /// 138000, 5170209, + /// 138001, 5170228, + /// \param featureId2OsmIdsPath path to the text file. + /// \note Most restrictions consist of type and two linear(road) features. + /// \note For the time being only line-point-line restritions are supported. + bool ParseFeatureId2OsmIdsMapping(string const & featureId2OsmIdsPath); + + /// \brief Parses comma separated text file with line in following format: + /// , , , and so on + /// For example: + /// Only, 335049632, 49356687, + /// No, 157616940, 157616940, + /// No, 157616940, 157617107, + /// \param featureId2OsmIdsPath path to the text file. + bool ParseRestrictions(string const & restrictionPath); + + /// \brief Sets feature id for all restrictions in |m_restrictions|. + void ComposeRestrictions(); /// \brief removes all restriction with incorrect feature id. /// \note The method should be called after ComposeRestrictions(). void RemoveInvalidRestrictions(); - /// \brief Sets feature id for all restrictions in |m_restrictions|. - void ComposeRestrictions(); + /// \brief Adds feature id and corresponding vector of |osmIds| to |m_osmId2FeatureId|. + /// \note One feature id (|featureId|) may correspond to several osm ids (|osmIds|). + void AddFeatureId(FeatureId featureId, vector const & osmIds); - /// \brief Adds a restriction (vector of osm::Id). - /// \param links is osm ids of restriction links + /// \brief Adds a restriction (vector of osm id). /// \param type is a type of restriction + /// \param osmIds is osm ids of restriction links /// \note This method should be called to add a restriction when feature ids of the restriction /// are unknown. The feature ids should be set later with a call of |SetFeatureId(...)| method. - void AddRestriction(vector const & links, Type type); + void AddRestriction(Type type, vector const & osmIds); - mutex m_mutex; vector m_restrictions; - vector> m_restrictionIndex; + vector> m_restrictionIndex; - unordered_multimap m_osmIds2FeatureId; + unordered_multimap m_osmIds2FeatureId; }; string ToString(RestrictionCollector::Type const & type); -bool FromString(string const & str, RestrictionCollector::Type & type); +bool FromString(string str, RestrictionCollector::Type & type); string DebugPrint(RestrictionCollector::Type const & type); +string DebugPrint(RestrictionCollector::Index const & index); string DebugPrint(RestrictionCollector::Restriction const & restriction);