From 59e7eaa61338b9fae3ae502c115f768fbf3bdfd9 Mon Sep 17 00:00:00 2001 From: Maksim Andrianov Date: Tue, 1 Oct 2019 15:30:04 +0300 Subject: [PATCH] [generator] Added complex_generator --- base/stl_helpers.hpp | 18 + generator/CMakeLists.txt | 9 + generator/complex_generator/CMakeLists.txt | 47 +++ .../complex_generator/complex_generator.cpp | 109 ++++++ generator/feature_sorter.cpp | 7 +- generator/filter_complex.cpp | 25 ++ generator/filter_complex.hpp | 18 + .../final_processor_intermediate_mwm.cpp | 78 ++++- .../final_processor_intermediate_mwm.hpp | 32 +- generator/gen_mwm_info.cpp | 6 + generator/gen_mwm_info.hpp | 4 + generator/generator_tool/generator_tool.cpp | 48 +-- generator/hierarchy.cpp | 325 ++++++++++++++++++ generator/hierarchy.hpp | 174 ++++++++++ generator/place_node.hpp | 35 +- generator/processor_complex.cpp | 57 +++ generator/processor_complex.hpp | 44 +++ generator/processor_factory.hpp | 6 +- generator/processor_interface.hpp | 2 + generator/translator_complex.cpp | 41 +++ generator/translator_complex.hpp | 32 ++ generator/translator_factory.hpp | 4 + generator/translator_interface.hpp | 2 + generator/utils.cpp | 46 +++ generator/utils.hpp | 25 +- indexer/ftypes_matcher.cpp | 55 +-- indexer/ftypes_matcher.hpp | 18 +- 27 files changed, 1145 insertions(+), 122 deletions(-) create mode 100644 generator/complex_generator/CMakeLists.txt create mode 100644 generator/complex_generator/complex_generator.cpp create mode 100644 generator/filter_complex.cpp create mode 100644 generator/filter_complex.hpp create mode 100644 generator/hierarchy.cpp create mode 100644 generator/hierarchy.hpp create mode 100644 generator/processor_complex.cpp create mode 100644 generator/processor_complex.hpp create mode 100644 generator/translator_complex.cpp create mode 100644 generator/translator_complex.hpp diff --git a/base/stl_helpers.hpp b/base/stl_helpers.hpp index 0cef78475a..d21cc6b70c 100644 --- a/base/stl_helpers.hpp +++ b/base/stl_helpers.hpp @@ -130,6 +130,24 @@ void EraseIf(Cont & c, Fn && fn) c.erase(remove_if(c.begin(), c.end(), std::forward(fn)), c.end()); } +template +bool AllOf(Cont && c, Fn && fn) +{ + return std::all_of(c.cbegin(), c.cend(), std::forward(fn)); +} + +template +bool AnyOf(Cont && c, Fn && fn) +{ + return std::any_of(c.cbegin(), c.cend(), std::forward(fn)); +} + +template +decltype(auto) FindIf(Cont && c, Fn && fn) +{ + return std::find_if(c.begin(), c.end(), std::forward(fn)); +} + // Creates a comparer being able to compare two instances of class C // (given by reference or pointer) by a field or const method of C. // For example, to create comparer that is able to compare pairs of diff --git a/generator/CMakeLists.txt b/generator/CMakeLists.txt index 502caa08df..7e759bd037 100644 --- a/generator/CMakeLists.txt +++ b/generator/CMakeLists.txt @@ -78,6 +78,8 @@ set( filter_roads.hpp filter_collection.cpp filter_collection.hpp + filter_complex.cpp + filter_complex.hpp filter_elements.cpp filter_elements.hpp filter_interface.hpp @@ -91,6 +93,8 @@ set( gen_mwm_info.hpp generate_info.hpp geometry_holder.hpp + hierarchy.cpp + hierarchy.hpp holes.cpp holes.hpp intermediate_data.cpp @@ -130,6 +134,8 @@ set( processor_booking.hpp processor_coastline.cpp processor_coastline.hpp + processor_complex.cpp + processor_complex.hpp processor_country.cpp processor_country.hpp processor_factory.hpp @@ -191,6 +197,8 @@ set( translator_coastline.hpp translator_collection.cpp translator_collection.hpp + translator_complex.cpp + translator_complex.hpp translator_country.cpp translator_country.hpp translator_factory.hpp @@ -225,6 +233,7 @@ omim_add_test_subdirectory(generator_tests) omim_add_test_subdirectory(generator_integration_tests) add_subdirectory(generator_tool) +add_subdirectory(complex_generator) add_subdirectory(booking_quality_check) add_subdirectory(extract_addr) add_subdirectory(feature_segments_checker) diff --git a/generator/complex_generator/CMakeLists.txt b/generator/complex_generator/CMakeLists.txt new file mode 100644 index 0000000000..78ff4b6e04 --- /dev/null +++ b/generator/complex_generator/CMakeLists.txt @@ -0,0 +1,47 @@ +project(complex_generator) + +include_directories(${OMIM_ROOT}/3party/gflags/src) + +set(SRC complex_generator.cpp) + +omim_add_executable(${PROJECT_NAME} ${SRC}) + +omim_link_libraries( + ${PROJECT_NAME} + generator + routing + traffic + routing_common + descriptions + transit + ugc + search + storage + editor + indexer + mwm_diff + platform + geometry + coding + base + opening_hours + freetype + expat + icu + jansson + protobuf + bsdiff + stats_client + minizip + succinct + pugixml + tess2 + gflags + oauthcpp + sqlite3 + ${CMAKE_DL_LIBS} + ${LIBZ} +) + +link_qt5_core(${PROJECT_NAME}) +link_qt5_network(${PROJECT_NAME}) diff --git a/generator/complex_generator/complex_generator.cpp b/generator/complex_generator/complex_generator.cpp new file mode 100644 index 0000000000..ee2ca24590 --- /dev/null +++ b/generator/complex_generator/complex_generator.cpp @@ -0,0 +1,109 @@ +// This is a program that generates complexes on the basis of the last generation of maps. +// Complexes are a hierarchy of interesting geographical features. +// For the program to work correctly, you need to have in your file system: +// top_directory +// |_ planet.o5m +// |_maps_build +// |_190223(for example 2019 Feb. 23rd) +// |_intermediate_data +// |_osm2ft +// +// It's easy if you use maps_generator. For example: +// +// $ python -m maps_generator --skip="coastline" --countries="Russia_Moscow" +// +// $ ./complex_generator --maps_build_path=path/to/maps_build \ +// --user_resource_path=path/to/omim/data --output=output.txt + +#include "generator/final_processor_intermediate_mwm.hpp" +#include "generator/generate_info.hpp" +#include "generator/hierarchy.hpp" +#include "generator/intermediate_data.hpp" +#include "generator/processor_factory.hpp" +#include "generator/raw_generator.hpp" +#include "generator/translator_factory.hpp" +#include "generator/utils.hpp" + +#include "indexer/classificator_loader.hpp" +#include "indexer/map_style_reader.hpp" + +#include "platform/platform.hpp" + +#include "coding/endianness.hpp" + +#include "base/assert.hpp" +#include "base/exception.hpp" + +#include +#include +#include +#include + +#include "build_version.hpp" +#include "defines.hpp" + +#include "3party/gflags/src/gflags/gflags.h" + +DEFINE_string(node_storage, "map", + "Type of storage for intermediate points representation. Available: raw, map, mem."); +DEFINE_string(user_resource_path, "", "User defined resource path for classificator.txt and etc."); +DEFINE_string(maps_build_path, "", + "Directory of any of the previous map generations. It is assumed that it will " + "contain a directory with mwm(for example 190423) and a directory with mappings from " + "osm is to a feature id."); +DEFINE_string(output, "", "Output filename"); +DEFINE_bool(debug, false, "Debug mode."); + +MAIN_WITH_ERROR_HANDLING([](int argc, char ** argv) { + CHECK(IsLittleEndian(), ("Only little-endian architectures are supported.")); + + google::SetUsageMessage( + "complex_generator is a program that generates complexes on the basis of " + "the last generation of maps. Complexes are a hierarchy of interesting " + "geographical features."); + google::SetVersionString(std::to_string(omim::build_version::git::kTimestamp) + " " + + omim::build_version::git::kHash); + google::ParseCommandLineFlags(&argc, &argv, true); + + Platform & pl = GetPlatform(); + auto threadsCount = pl.CpuCores(); + CHECK(!FLAGS_user_resource_path.empty(), ()); + pl.SetResourceDir(FLAGS_user_resource_path); + classificator::Load(); + + feature::GenerateInfo genInfo; + genInfo.m_osmFileName = base::JoinPath(FLAGS_maps_build_path, "..", "planet.o5m"); + genInfo.SetOsmFileType("o5m"); + genInfo.SetNodeStorageType(FLAGS_node_storage); + genInfo.m_intermediateDir = base::JoinPath(FLAGS_maps_build_path, "intermediate_data"); + genInfo.m_tmpDir = base::JoinPath(FLAGS_maps_build_path, "complex", "tmp"); + CHECK(Platform::MkDirRecursively(genInfo.m_tmpDir), ()); + // Directory FLAGS_maps_build_path must contain 'osm2ft' directory with *.mwm.osm2ft + auto const osm2FtPath = base::JoinPath(FLAGS_maps_build_path, "osm2ft"); + // Find directory with *.mwm. Directory FLAGS_maps_build_path must contain directory with *.mwm, + // which name must contain six digits. + Platform::FilesList files; + pl.GetFilesByRegExp(FLAGS_maps_build_path, "[0-9]{6}", files); + CHECK_EQUAL(files.size(), 1, ()); + auto const mwmPath = base::JoinPath(FLAGS_maps_build_path, files[0]); + + generator::RawGenerator rawGenerator(genInfo, threadsCount); + auto processor = CreateProcessor(generator::ProcessorType::Complex, rawGenerator.GetQueue(), + genInfo.m_intermediateDir, "" /* layerLogFilename */, + false /* haveBordersForWholeWorld */); + auto const cache = std::make_shared(genInfo); + auto translator = + CreateTranslator(generator::TranslatorType::Complex, processor, cache); + auto finalProcessor = std::make_shared( + genInfo.m_tmpDir, FLAGS_output, threadsCount); + finalProcessor->SetMwmAndFt2OsmPath(mwmPath, osm2FtPath); + if (FLAGS_debug) + { + finalProcessor->SetPrintFunction( + static_cast( + generator::hierarchy::DebugPrint)); + } + rawGenerator.GenerateCustom(translator, finalProcessor); + CHECK(rawGenerator.Execute(), ()); + return EXIT_SUCCESS; +}); diff --git a/generator/feature_sorter.cpp b/generator/feature_sorter.cpp index 418e436d68..801658c8b1 100644 --- a/generator/feature_sorter.cpp +++ b/generator/feature_sorter.cpp @@ -238,11 +238,8 @@ public: fb.GetMetadata().Serialize(*w); } - if (!fb.GetOsmIds().empty()) - { - generator::CompositeId const id(fb.GetMostGenericOsmId(), fb.GetFirstOsmId()); - m_osm2ft.AddIds(id, featureId); - } + if (fb.HasOsmIds()) + m_osm2ft.AddIds(generator::MakeCompositeId(fb), featureId); }; return featureId; } diff --git a/generator/filter_complex.cpp b/generator/filter_complex.cpp new file mode 100644 index 0000000000..64ba39a1fa --- /dev/null +++ b/generator/filter_complex.cpp @@ -0,0 +1,25 @@ +#include "generator/filter_complex.hpp" + +#include "generator/feature_builder.hpp" + +#include "indexer/ftypes_matcher.hpp" + +namespace generator +{ +std::shared_ptr FilterComplex::Clone() const +{ + return std::make_shared(); +} + +bool FilterComplex::IsAccepted(feature::FeatureBuilder const & fb) +{ + if (!fb.IsArea() && !fb.IsPoint()) + return false; + + auto const & eatChecker = ftypes::IsEatChecker::Instance(); + auto const & attractChecker = ftypes::AttractionsChecker::Instance(); + auto const & airportChecker = ftypes::IsAirportChecker::Instance(); + auto const & ts = fb.GetTypes(); + return eatChecker(ts) || attractChecker(ts) || airportChecker(ts); +} +} // namespace generator diff --git a/generator/filter_complex.hpp b/generator/filter_complex.hpp new file mode 100644 index 0000000000..4983ac94c6 --- /dev/null +++ b/generator/filter_complex.hpp @@ -0,0 +1,18 @@ +#pragma once + +#include "generator/filter_interface.hpp" + +#include + +namespace generator +{ +// The filter will leave only elements for complex(checks fb is attraction, eat or airport). +class FilterComplex : public FilterInterface +{ +public: + // FilterInterface overrides: + std::shared_ptr Clone() const override; + + bool IsAccepted(feature::FeatureBuilder const & fb) override; +}; +} // namespace generator diff --git a/generator/final_processor_intermediate_mwm.cpp b/generator/final_processor_intermediate_mwm.cpp index b3e1018540..36d78e6477 100644 --- a/generator/final_processor_intermediate_mwm.cpp +++ b/generator/final_processor_intermediate_mwm.cpp @@ -9,8 +9,10 @@ #include "generator/place_processor.hpp" #include "generator/promo_catalog_cities.hpp" #include "generator/type_helper.hpp" +#include "generator/utils.hpp" #include "indexer/classificator.hpp" +#include "indexer/feature_algo.hpp" #include "platform/platform.hpp" @@ -380,7 +382,7 @@ void CountryFinalProcessor::ProcessCoastline() FeatureBuilderWriter<> collector(m_worldCoastsFilename); for (size_t i = 0; i < fbs.size(); ++i) { - fbs[i].AddName("default", strings::JoinStrings(affiliations[i], ";")); + fbs[i].AddName("default", strings::JoinStrings(affiliations[i], ';')); collector.Write(fbs[i]); } } @@ -508,4 +510,78 @@ void CoastlineFinalProcessor::Process() LOG(LINFO, ("Total features:", totalFeatures, "total polygons:", totalPolygons, "total points:", totalPoints)); } + +ComplexFinalProcessor::ComplexFinalProcessor(std::string const & mwmTmpPath, + std::string const & outFilename, size_t threadsCount) + : FinalProcessorIntermediateMwmInterface(FinalProcessorPriority::Complex) + , m_mwmTmpPath(mwmTmpPath) + , m_outFilename(outFilename) + , m_threadsCount(threadsCount) +{ +} + +void ComplexFinalProcessor::SetMwmAndFt2OsmPath(std::string const & mwmPath, + std::string const & osm2ftPath) +{ + m_mwmPath = mwmPath; + m_osm2ftPath = osm2ftPath; +} + +void ComplexFinalProcessor::SetPrintFunction(hierarchy::PrintFunction const & printFunction) +{ + m_printFunction = printFunction; +} + +std::shared_ptr ComplexFinalProcessor::CreateEnricher( + std::string const & countryName) const +{ + if (m_osm2ftPath.empty() || m_mwmPath.empty()) + return {}; + + return std::make_shared( + base::JoinPath(m_osm2ftPath, countryName + DATA_FILE_EXTENSION OSM2FEATURE_FILE_EXTENSION), + base::JoinPath(m_mwmPath, countryName + DATA_FILE_EXTENSION)); +} + +void ComplexFinalProcessor::Process() +{ + ThreadPool pool(m_threadsCount); + std::vector>> futures; + ForEachCountry(m_mwmTmpPath, [&](auto const & filename) { + auto future = pool.Submit([&, filename]() { + auto countryName = filename; + strings::ReplaceLast(countryName, DATA_FILE_EXTENSION_TMP, ""); + + hierarchy::HierarchyBuilder builder(base::JoinPath(m_mwmTmpPath, filename)); + builder.SetGetMainTypeFunction(hierarchy::popularity::GetMainType); + builder.SetGetNameFunction(hierarchy::popularity::GetName); + auto nodes = builder.Build(); + + auto const enricher = CreateEnricher(countryName); + hierarchy::HierarchyLinesBuilder linesBuilder(std::move(nodes)); + linesBuilder.SetHierarchyLineEnricher(enricher); + linesBuilder.SetCountryName(countryName); + linesBuilder.SetGetMainTypeFunction(hierarchy::popularity::GetMainType); + linesBuilder.SetGetNameFunction(hierarchy::popularity::GetName); + return linesBuilder.GetHierarchyLines(); + }); + futures.emplace_back(std::move(future)); + }); + std::vector allLines; + for (auto & f : futures) + { + auto const lines = f.get(); + allLines.insert(std::cend(allLines), std::cbegin(lines), std::cend(lines)); + } + WriteLines(allLines); +} + +void ComplexFinalProcessor::WriteLines(std::vector const & lines) +{ + std::ofstream stream; + stream.exceptions(std::fstream::failbit | std::fstream::badbit); + stream.open(m_outFilename); + for (auto const & line : lines) + stream << m_printFunction(line) << '\n'; +} } // namespace generator diff --git a/generator/final_processor_intermediate_mwm.hpp b/generator/final_processor_intermediate_mwm.hpp index a2dfb08f12..e041e4dc1e 100644 --- a/generator/final_processor_intermediate_mwm.hpp +++ b/generator/final_processor_intermediate_mwm.hpp @@ -2,6 +2,7 @@ #include "generator/coastlines_generator.hpp" #include "generator/feature_generator.hpp" +#include "generator/hierarchy.hpp" #include "generator/world_map_generator.hpp" #include @@ -13,7 +14,8 @@ namespace generator enum class FinalProcessorPriority : uint8_t { CountriesOrWorld = 1, - WorldCoasts = 2 + WorldCoasts = 2, + Complex = 3 }; // Classes that inherit this interface implement the final stage of intermediate mwm processing. @@ -120,4 +122,32 @@ private: std::string m_coastlineRawGeomFilename; CoastlineFeaturesGenerator m_generator; }; + +// Class ComplexFinalProcessor generates hierarchies for each previously filtered mwm.tmp file. +// Warning: If the border separates the complex, then a situation is possible in which two logically +// identical complexes are generated, but with different representations. +class ComplexFinalProcessor : public FinalProcessorIntermediateMwmInterface +{ +public: + ComplexFinalProcessor(std::string const & mwmTmpPath, std::string const & outFilename, + size_t threadsCount); + + void SetMwmAndFt2OsmPath(std::string const & mwmPath, std::string const & osm2ftPath); + void SetPrintFunction(hierarchy::PrintFunction const & printFunction); + + // FinalProcessorIntermediateMwmInterface overrides: + void Process() override; + +private: + std::shared_ptr CreateEnricher( + std::string const & countryName) const; + void WriteLines(std::vector const & lines); + + hierarchy::PrintFunction m_printFunction = hierarchy::PrintDefault; + std::string m_mwmTmpPath; + std::string m_outFilename; + std::string m_mwmPath; + std::string m_osm2ftPath; + size_t m_threadsCount; +}; } // namespace generator diff --git a/generator/gen_mwm_info.cpp b/generator/gen_mwm_info.cpp index 818c71f18f..1b5ec7c724 100644 --- a/generator/gen_mwm_info.cpp +++ b/generator/gen_mwm_info.cpp @@ -46,6 +46,12 @@ std::string CompositeId::ToString() const return stream.str(); } +CompositeId MakeCompositeId(feature::FeatureBuilder const & fb) +{ + CHECK(fb.HasOsmIds(), (fb)); + return CompositeId(fb.GetMostGenericOsmId(), fb.GetFirstOsmId()); +} + std::string DebugPrint(CompositeId const & id) { return DebugPrint(id.m_mainId) + "|" + DebugPrint(id.m_additionalId); diff --git a/generator/gen_mwm_info.hpp b/generator/gen_mwm_info.hpp index 834ca28bd9..476248fac3 100644 --- a/generator/gen_mwm_info.hpp +++ b/generator/gen_mwm_info.hpp @@ -1,5 +1,7 @@ #pragma once +#include "generator/feature_builder.hpp" + #include "coding/file_reader.hpp" #include "coding/read_write_utils.hpp" #include "coding/write_to_sink.hpp" @@ -37,6 +39,8 @@ struct CompositeId base::GeoObjectId m_additionalId; }; +CompositeId MakeCompositeId(feature::FeatureBuilder const & fb); + std::string DebugPrint(CompositeId const & id); } // namespace generator diff --git a/generator/generator_tool/generator_tool.cpp b/generator/generator_tool/generator_tool.cpp index 34aef471b0..d0cd24be9c 100644 --- a/generator/generator_tool/generator_tool.cpp +++ b/generator/generator_tool/generator_tool.cpp @@ -31,6 +31,7 @@ #include "generator/translator_factory.hpp" #include "generator/ugc_section_builder.hpp" #include "generator/unpack_mwm.hpp" +#include "generator/utils.hpp" #include "generator/wiki_url_dumper.hpp" #include "routing/cross_mwm_ids.hpp" @@ -63,9 +64,6 @@ #include #include -#define BOOST_STACKTRACE_GNU_SOURCE_NOT_REQUIRED -#include - #include "build_version.hpp" #include "defines.hpp" @@ -197,7 +195,7 @@ DEFINE_bool(verbose, false, "Provide more detailed output."); using namespace generator; -int GeneratorToolMain(int argc, char ** argv) +MAIN_WITH_ERROR_HANDLING([](int argc, char ** argv) { CHECK(IsLittleEndian(), ("Only little-endian architectures are supported.")); @@ -574,43 +572,5 @@ int GeneratorToolMain(int argc, char ** argv) if (FLAGS_check_mwm) check_model::ReadFeatures(datFile); - return 0; -} - -void ErrorHandler(int signum) -{ - // Avoid recursive calls. - signal(signum, SIG_DFL); - - // If there was an exception, then we will print the message. - try - { - if (auto const eptr = current_exception()) - rethrow_exception(eptr); - } - catch (RootException const & e) - { - cerr << "Core exception: " << e.Msg() << "\n"; - } - catch (exception const & e) - { - cerr << "Std exception: " << e.what() << "\n"; - } - catch (...) - { - cerr << "Unknown exception.\n"; - } - - // Print stack stack. - cerr << boost::stacktrace::stacktrace(); - // We raise the signal SIGABRT, so that there would be an opportunity to make a core dump. - raise(SIGABRT); -} - -int main(int argc, char ** argv) -{ - signal(SIGABRT, ErrorHandler); - signal(SIGSEGV, ErrorHandler); - - return GeneratorToolMain(argc, argv); -} + return EXIT_SUCCESS; +}); diff --git a/generator/hierarchy.cpp b/generator/hierarchy.cpp new file mode 100644 index 0000000000..4b87cc1dc1 --- /dev/null +++ b/generator/hierarchy.cpp @@ -0,0 +1,325 @@ +#include "generator/hierarchy.hpp" + +#include "generator/boost_helpers.hpp" +#include "generator/place_processor.hpp" + +#include "indexer/classificator.hpp" +#include "indexer/feature_algo.hpp" +#include "indexer/feature_utils.hpp" + +#include "geometry/mercator.hpp" +#include "geometry/rect2d.hpp" + +#include "base/stl_helpers.hpp" +#include "base/string_utils.hpp" + +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include + +BOOST_GEOMETRY_REGISTER_POINT_2D(m2::PointD, double, boost::geometry::cs::cartesian, x, y); +BOOST_GEOMETRY_REGISTER_RING(std::vector); + +using namespace feature; + +namespace generator +{ +namespace hierarchy +{ +namespace +{ +// GetRussianName returns a russian feature name if it's possible. +// Otherwise, GetRussianName function returns a name that GetReadableName returns. +std::string GetRussianName(StringUtf8Multilang const & str) +{ + auto const deviceLang = StringUtf8Multilang::GetLangIndex("ru"); + std::string result; + GetReadableName({} /* regionData */, str, deviceLang, false /* allowTranslit */, result); + for (auto const & ch : {';', '\n', '\t'}) + std::replace(std::begin(result), std::end(result), ch, ','); + return result; +} +} // namespace + +uint32_t GetTypeDefault(FeatureParams::Types const &) { return ftype::GetEmptyValue(); } + +std::string GetNameDefault(StringUtf8Multilang const &) { return {}; } + +std::string PrintDefault(HierarchyLine const &) { return {}; } + +HierarchyPlace::HierarchyPlace(FeatureBuilder const & fb) + : m_id(MakeCompositeId(fb)) + , m_name(fb.GetMultilangName()) + , m_types(fb.GetTypes()) + , m_rect(fb.GetLimitRect()) + , m_center(fb.GetKeyPoint()) +{ + if (fb.IsPoint()) + { + m_isPoint = true; + } + else if (fb.IsArea()) + { + m_polygon = fb.GetOuterGeometry(); + boost::geometry::correct(m_polygon); + m_area = boost::geometry::area(m_polygon); + } +} + +bool HierarchyPlace::Contains(HierarchyPlace const & smaller) const +{ + if (IsPoint()) + return false; + + if (smaller.IsPoint()) + return Contains(smaller.GetCenter()); + + return m_rect.IsRectInside(smaller.GetLimitRect()) && + boost::geometry::covered_by(smaller.m_polygon, m_polygon); +} + +bool HierarchyPlace::Contains(m2::PointD const & point) const +{ + return boost::geometry::covered_by(point, m_polygon); +} + +bool HierarchyPlace::IsEqualGeometry(HierarchyPlace const & other) const +{ + return IsPoint() ? boost::geometry::equals(m_center, other.m_center) + : boost::geometry::equals(m_polygon, other.m_polygon); +} + +HierarchyLinker::HierarchyLinker(Node::PtrList && nodes) + : m_nodes(std::move(nodes)), m_tree(MakeTree4d(m_nodes)) +{ +} + +// static +HierarchyLinker::Tree4d HierarchyLinker::MakeTree4d(Node::PtrList const & nodes) +{ + Tree4d tree; + for (auto const & n : nodes) + tree.Add(n, n->GetData().GetLimitRect()); + return tree; +} + +HierarchyLinker::Node::Ptr HierarchyLinker::FindPlaceParent(HierarchyPlace const & place) +{ + Node::Ptr parent = nullptr; + auto minArea = std::numeric_limits::max(); + auto const point = place.GetCenter(); + m_tree.ForEachInRect({point, point}, [&](auto const & candidatePtr) { + auto const & candidate = candidatePtr->GetData(); + if (place.GetCompositeId() == candidate.GetCompositeId()) + return; + // Sometimes there can be two places with the same geometry. We must compare their ids + // to avoid cyclic connections. + if (place.IsEqualGeometry(candidate)) + { + if (place.GetCompositeId() < candidate.GetCompositeId()) + parent = candidatePtr; + } + else if (candidate.GetArea() < minArea && candidate.Contains(place)) + { + parent = candidatePtr; + minArea = candidate.GetArea(); + } + }); + return parent; +} + +HierarchyLinker::Node::PtrList HierarchyLinker::Link() +{ + for (auto & node : m_nodes) + { + auto const & place = node->GetData(); + auto const parentPlace = FindPlaceParent(place); + if (!parentPlace) + continue; + + parentPlace->AddChild(node); + node->SetParent(parentPlace); + } + return m_nodes; +} + +HierarchyBuilder::HierarchyBuilder(std::string const & dataFilename) + : m_dataFullFilename(dataFilename) +{ +} + +void HierarchyBuilder::SetGetMainTypeFunction(GetMainType const & getMainType) +{ + m_getMainType = getMainType; +} + +void HierarchyBuilder::SetGetNameFunction(GetName const & getName) { m_getName = getName; } + +std::vector HierarchyBuilder::ReadFeatures( + std::string const & dataFilename) +{ + std::vector fbs; + ForEachFromDatRawFormat( + dataFilename, [&](FeatureBuilder const & fb, uint64_t /* currPos */) { + if (m_getMainType(fb.GetTypes()) != ftype::GetEmptyValue() && + !m_getName(fb.GetMultilangName()).empty() && !fb.GetOsmIds().empty() && + (fb.IsPoint() || fb.IsArea())) + { + fbs.emplace_back(fb); + } + }); + return fbs; +} + +HierarchyBuilder::Node::PtrList HierarchyBuilder::Build() +{ + auto const fbs = ReadFeatures(m_dataFullFilename); + Node::PtrList places; + places.reserve(fbs.size()); + std::transform(std::cbegin(fbs), std::cend(fbs), std::back_inserter(places), + [](auto const & fb) { return std::make_shared(HierarchyPlace(fb)); }); + return HierarchyLinker(std::move(places)).Link(); +} + +HierarchyLineEnricher::HierarchyLineEnricher(std::string const & osm2FtIdsPath, + std::string const & countryFullPath) + : m_featureGetter(countryFullPath) +{ + CHECK(m_osm2FtIds.ReadFromFile(osm2FtIdsPath), (osm2FtIdsPath)); +} + +boost::optional HierarchyLineEnricher::GetFeatureCenter(CompositeId const & id) const +{ + auto const optId = m_osm2FtIds.GetFeatureId(id); + if (!optId) + return {}; + + auto const ftPtr = m_featureGetter.GetFeatureByIndex(*optId); + return ftPtr ? feature::GetCenter(*ftPtr) : boost::optional(); +} + +std::string DebugPrint(HierarchyLine const & line) +{ + std::stringstream stream; + stream << std::fixed << std::setprecision(7); + stream << DebugPrint(line.m_id) << ';'; + if (line.m_parentId) + stream << DebugPrint(*line.m_parentId); + stream << ';'; + stream << line.m_depth << ';'; + stream << line.m_center.x << ';'; + stream << line.m_center.y << ';'; + stream << classif().GetReadableObjectName(line.m_type) << ';'; + stream << line.m_name << ';'; + stream << line.m_countryName; + return stream.str(); +} + +HierarchyLinesBuilder::HierarchyLinesBuilder(HierarchyBuilder::Node::PtrList && nodes) + : m_nodes(std::move(nodes)) +{ +} + +void HierarchyLinesBuilder::SetGetMainTypeFunction(GetMainType const & getMainType) +{ + m_getMainType = getMainType; +} + +void HierarchyLinesBuilder::SetGetNameFunction(GetName const & getName) { m_getName = getName; } + +void HierarchyLinesBuilder::SetCountryName(std::string const & name) { m_countryName = name; } + +void HierarchyLinesBuilder::SetHierarchyLineEnricher( + std::shared_ptr const & enricher) +{ + m_enricher = enricher; +} + +std::vector HierarchyLinesBuilder::GetHierarchyLines() +{ + std::vector lines; + lines.reserve(m_nodes.size()); + std::transform(std::cbegin(m_nodes), std::cend(m_nodes), std::back_inserter(lines), + [&](auto const & n) { return Transform(n); }); + return lines; +} + +m2::PointD HierarchyLinesBuilder::GetCenter(HierarchyBuilder::Node::Ptr const & node) +{ + auto const & data = node->GetData(); + if (!m_enricher) + return data.GetCenter(); + + auto const optCenter = m_enricher->GetFeatureCenter(data.GetCompositeId()); + return optCenter ? *optCenter : data.GetCenter(); +} + +HierarchyLine HierarchyLinesBuilder::Transform(HierarchyBuilder::Node::Ptr const & node) +{ + HierarchyLine line; + auto const & data = node->GetData(); + line.m_id = data.GetCompositeId(); + auto const parent = node->GetParent(); + if (parent) + line.m_parentId = parent->GetData().GetCompositeId(); + + line.m_countryName = m_countryName; + line.m_depth = GetDepth(node); + line.m_name = m_getName(data.GetName()); + line.m_type = m_getMainType(data.GetTypes()); + line.m_center = GetCenter(node); + return line; +} + +namespace popularity +{ +uint32_t GetMainType(FeatureParams::Types const & types) +{ + auto const & airportChecker = ftypes::IsAirportChecker::Instance(); + auto it = base::FindIf(types, airportChecker); + if (it != std::cend(types)) + return *it; + + auto const & attractChecker = ftypes::AttractionsChecker::Instance(); + auto const type = attractChecker.GetBestType(types); + if (type != ftype::GetEmptyValue()) + return type; + + auto const & eatChecker = ftypes::IsEatChecker::Instance(); + it = base::FindIf(types, eatChecker); + return it != std::cend(types) ? *it : ftype::GetEmptyValue(); +} + +std::string GetName(StringUtf8Multilang const & str) { return GetRussianName(str); } + +std::string Print(HierarchyLine const & line) +{ + std::stringstream stream; + stream << std::fixed << std::setprecision(7); + stream << line.m_id.m_mainId.GetEncodedId() << '|' << line.m_id.m_additionalId.GetEncodedId() + << ';'; + if (line.m_parentId) + { + auto const parentId = *line.m_parentId; + stream << parentId.m_mainId.GetEncodedId() << '|' << parentId.m_additionalId.GetEncodedId() + << ';'; + } + stream << ';'; + stream << line.m_center.x << ';'; + stream << line.m_center.y << ';'; + stream << classif().GetReadableObjectName(line.m_type) << ';'; + stream << line.m_name; + return stream.str(); +} + +} // namespace popularity +} // namespace hierarchy +} // namespace generator diff --git a/generator/hierarchy.hpp b/generator/hierarchy.hpp new file mode 100644 index 0000000000..86c11e0726 --- /dev/null +++ b/generator/hierarchy.hpp @@ -0,0 +1,174 @@ +#pragma once + +#include "generator/feature_builder.hpp" +#include "generator/gen_mwm_info.hpp" +#include "generator/place_node.hpp" +#include "generator/platform_helpers.hpp" +#include "generator/utils.hpp" + +#include "indexer/classificator.hpp" +#include "indexer/ftypes_matcher.hpp" + +#include "geometry/point2d.hpp" +#include "geometry/tree4d.hpp" + +#include "base/assert.hpp" +#include "base/file_name_utils.hpp" +#include "base/geo_object_id.hpp" +#include "base/thread_pool_computational.hpp" + +#include +#include +#include +#include +#include +#include + +#include +#include + +namespace generator +{ +namespace hierarchy +{ +struct HierarchyLine; + +using GetMainType = std::function; +using GetName = std::function; +using PrintFunction = std::function; + +// These are dummy functions. +uint32_t GetTypeDefault(FeatureParams::Types const &); +std::string GetNameDefault(StringUtf8Multilang const &); +std::string PrintDefault(HierarchyLine const &); + +// The HierarchyPlace class is an abstraction of FeatureBuilder to build a hierarchy of objects. +// It allows you to work with the geometry of points and areas. +class HierarchyPlace +{ +public: + explicit HierarchyPlace(feature::FeatureBuilder const & fb); + + double GetArea() const { return m_area; } + CompositeId const & GetCompositeId() const { return m_id; } + StringUtf8Multilang const & GetName() const { return m_name; } + FeatureParams::Types const & GetTypes() const { return m_types; } + m2::RectD const & GetLimitRect() const { return m_rect; } + m2::PointD const & GetCenter() const { return m_center; } + bool IsPoint() const { return m_isPoint; } + + bool Contains(HierarchyPlace const & smaller) const; + bool IsEqualGeometry(HierarchyPlace const & other) const; + +private: + bool Contains(m2::PointD const & point) const; + + CompositeId m_id; + StringUtf8Multilang m_name; + feature::FeatureBuilder::PointSeq m_polygon; + FeatureParams::Types m_types; + m2::RectD m_rect; + double m_area = 0.0; + bool m_isPoint = false; + m2::PointD m_center; +}; + +// The HierarchyLinker class allows you to link nodes into a tree according to the hierarchy. +// The knowledge of geometry only is used here. +class HierarchyLinker +{ +public: + using Node = PlaceNode; + using Tree4d = m4::Tree; + + explicit HierarchyLinker(Node::PtrList && nodes); + + Node::PtrList Link(); + +private: + static Tree4d MakeTree4d(Node::PtrList const & nodes); + + Node::Ptr FindPlaceParent(HierarchyPlace const & place); + + Node::PtrList m_nodes; + Tree4d m_tree; +}; + +// HierarchyBuilder class filters input elements and builds hierarchy from file *.mwm.tmp. +class HierarchyBuilder +{ +public: + using Node = HierarchyLinker::Node; + + explicit HierarchyBuilder(std::string const & dataFilename); + + void SetGetMainTypeFunction(GetMainType const & getMainType); + void SetGetNameFunction(GetName const & getName); + + Node::PtrList Build(); + +protected: + std::vector ReadFeatures(std::string const & dataFilename); + + std::string m_dataFullFilename; + GetMainType m_getMainType = GetTypeDefault; + GetName m_getName = GetNameDefault; +}; + +class HierarchyLineEnricher +{ +public: + HierarchyLineEnricher(std::string const & osm2FtIdsPath, std::string const & countryFullPath); + + boost::optional GetFeatureCenter(CompositeId const & id) const; + +private: + OsmID2FeatureID m_osm2FtIds; + FeatureGetter m_featureGetter; +}; + +// Intermediate view for hierarchy node. +struct HierarchyLine +{ + CompositeId m_id; + boost::optional m_parentId; + size_t m_depth = 0; + std::string m_name; + std::string m_countryName; + m2::PointD m_center; + uint32_t m_type = ftype::GetEmptyValue(); +}; + +std::string DebugPrint(HierarchyLine const & line); + +class HierarchyLinesBuilder +{ +public: + HierarchyLinesBuilder(HierarchyBuilder::Node::PtrList && nodes); + + void SetGetMainTypeFunction(GetMainType const & getMainType); + void SetGetNameFunction(GetName const & getName); + void SetCountryName(std::string const & name); + void SetHierarchyLineEnricher(std::shared_ptr const & enricher); + + std::vector GetHierarchyLines(); + +private: + m2::PointD GetCenter(HierarchyBuilder::Node::Ptr const & node); + HierarchyLine Transform(HierarchyBuilder::Node::Ptr const & node); + + HierarchyBuilder::Node::PtrList m_nodes; + GetMainType m_getMainType = GetTypeDefault; + GetName m_getName = GetNameDefault; + std::string m_countryName; + std::shared_ptr m_enricher; +}; + +namespace popularity +{ +uint32_t GetMainType(FeatureParams::Types const & types); +std::string GetName(StringUtf8Multilang const & str); +std::string Print(HierarchyLine const & line); +} // namespace popularity +} // namespace hierarchy +} // namespace generator diff --git a/generator/place_node.hpp b/generator/place_node.hpp index 8334f39fc3..902bc5e506 100644 --- a/generator/place_node.hpp +++ b/generator/place_node.hpp @@ -6,15 +6,30 @@ namespace generator { +template +class PlaceNode; + +namespace place_node_types +{ +template +using Ptr = std::shared_ptr>; + +template +using WeakPtr = std::weak_ptr>; + +template +using PtrList = std::vector>; +} // place_node_types + // This is a tree node, where each node can have many children. Now it is used to build a hierarchy // of places. template class PlaceNode { public: - using Ptr = std::shared_ptr; - using WeakPtr = std::weak_ptr; - using PtrList = std::vector; + using Ptr = place_node_types::Ptr; + using WeakPtr = place_node_types::WeakPtr; + using PtrList = place_node_types::PtrList; explicit PlaceNode(Data && data) : m_data(std::move(data)) {} @@ -38,8 +53,20 @@ private: WeakPtr m_parent; }; +template +size_t GetDepth(place_node_types::Ptr node) +{ + size_t depth = 0; + while (node) + { + node = node->GetParent(); + ++depth; + } + return depth; +} + template -void Visit(std::shared_ptr> const & tree, Visitor && visitor) +void Visit(place_node_types::Ptr const & tree, Visitor && visitor) { visitor(tree); for (auto const & subtree : tree->GetChildren()) diff --git a/generator/processor_complex.cpp b/generator/processor_complex.cpp new file mode 100644 index 0000000000..d00b388514 --- /dev/null +++ b/generator/processor_complex.cpp @@ -0,0 +1,57 @@ +#include "generator/processor_complex.hpp" + +#include "generator/feature_builder.hpp" +#include "generator/generate_info.hpp" + +#include "base/logging.hpp" + +#include + +namespace generator +{ +ProcessorComplex::ProcessorComplex(std::shared_ptr const & queue, + std::string const & bordersPath, + std::string const & layerLogFilename, + bool haveBordersForWholeWorld) + : m_bordersPath(bordersPath) + , m_layerLogFilename(layerLogFilename) + , m_queue(queue) + , m_haveBordersForWholeWorld(haveBordersForWholeWorld) +{ + m_processingChain = std::make_shared(); + auto affiliation = std::make_shared( + bordersPath, haveBordersForWholeWorld); + m_affiliationsLayer = + std::make_shared>(kAffiliationsBufferSize, affiliation, m_queue); + m_processingChain->Add(m_affiliationsLayer); +} + +std::shared_ptr ProcessorComplex::Clone() const +{ + return std::make_shared(m_queue, m_bordersPath, m_layerLogFilename, + m_haveBordersForWholeWorld); +} + +void ProcessorComplex::Process(feature::FeatureBuilder & feature) +{ + m_processingChain->Handle(feature); +} + +void ProcessorComplex::Finish() { m_affiliationsLayer->AddBufferToQueue(); } + +void ProcessorComplex::WriteDump() +{ + std::ofstream file; + file.exceptions(std::ios::failbit | std::ios::badbit); + file.open(m_layerLogFilename); + file << m_processingChain->GetAsStringRecursive(); + LOG(LINFO, ("Skipped elements were saved to", m_layerLogFilename)); +} + +void ProcessorComplex::Merge(FeatureProcessorInterface const & other) { other.MergeInto(*this); } + +void ProcessorComplex::MergeInto(ProcessorComplex & other) const +{ + other.m_processingChain->MergeChain(m_processingChain); +} +} // namespace generator diff --git a/generator/processor_complex.hpp b/generator/processor_complex.hpp new file mode 100644 index 0000000000..067baf25e6 --- /dev/null +++ b/generator/processor_complex.hpp @@ -0,0 +1,44 @@ +#pragma once + +#include "generator/feature_builder.hpp" +#include "generator/feature_processing_layers.hpp" +#include "generator/processor_interface.hpp" + +#include +#include + +namespace feature +{ +struct GenerateInfo; +} // namespace feature + +namespace generator +{ +// This class is the implementation of FeatureProcessorInterface for complexes. +class ProcessorComplex : public FeatureProcessorInterface +{ +public: + explicit ProcessorComplex(std::shared_ptr const & queue, + std::string const & bordersPath, std::string const & layerLogFilename, + bool haveBordersForWholeWorld); + + // FeatureProcessorInterface overrides: + std::shared_ptr Clone() const override; + + void Process(feature::FeatureBuilder & feature) override; + void Finish() override; + + void Merge(FeatureProcessorInterface const & other) override; + void MergeInto(ProcessorComplex & other) const override; + +private: + void WriteDump(); + + std::string m_bordersPath; + std::string m_layerLogFilename; + std::shared_ptr> m_affiliationsLayer; + std::shared_ptr m_queue; + std::shared_ptr m_processingChain; + bool m_haveBordersForWholeWorld; +}; +} // namespace generator diff --git a/generator/processor_factory.hpp b/generator/processor_factory.hpp index 4995c4046f..adff5237c0 100644 --- a/generator/processor_factory.hpp +++ b/generator/processor_factory.hpp @@ -3,6 +3,7 @@ #include "generator/factory_utils.hpp" #include "generator/processor_booking.hpp" #include "generator/processor_coastline.hpp" +#include "generator/processor_complex.hpp" #include "generator/processor_country.hpp" #include "generator/processor_interface.hpp" #include "generator/processor_noop.hpp" @@ -22,7 +23,8 @@ enum class ProcessorType Country, Coastline, World, - Noop + Noop, + Complex // Booking }; @@ -36,6 +38,8 @@ std::shared_ptr CreateProcessor(ProcessorType type, A case ProcessorType::Simple: return create(std::forward(args)...); case ProcessorType::World: return create(std::forward(args)...); case ProcessorType::Noop: return create(std::forward(args)...); + case ProcessorType::Complex: + return create(std::forward(args)...); } UNREACHABLE(); } diff --git a/generator/processor_interface.hpp b/generator/processor_interface.hpp index 5bb1ff5817..57a6a0a528 100644 --- a/generator/processor_interface.hpp +++ b/generator/processor_interface.hpp @@ -18,6 +18,7 @@ class ProcessorCountry; class ProcessorNoop; class ProcessorSimple; class ProcessorWorld; +class ProcessorComplex; // Implementing this interface allows an object to process FeatureBuilder objects and broadcast // them. @@ -39,6 +40,7 @@ public: virtual void MergeInto(ProcessorNoop &) const { FailIfMethodUnsupported(); } virtual void MergeInto(ProcessorSimple &) const { FailIfMethodUnsupported(); } virtual void MergeInto(ProcessorWorld &) const { FailIfMethodUnsupported(); } + virtual void MergeInto(ProcessorComplex &) const { FailIfMethodUnsupported(); } private: void FailIfMethodUnsupported() const { CHECK(false, ("This method is unsupported.")); } diff --git a/generator/translator_complex.cpp b/generator/translator_complex.cpp new file mode 100644 index 0000000000..64cc02c103 --- /dev/null +++ b/generator/translator_complex.cpp @@ -0,0 +1,41 @@ +#include "generator/translator_complex.hpp" + +#include "generator/collector_interface.hpp" +#include "generator/feature_maker.hpp" +#include "generator/filter_collection.hpp" +#include "generator/filter_complex.hpp" +#include "generator/filter_elements.hpp" +#include "generator/filter_planet.hpp" + +#include "generator/generate_info.hpp" +#include "generator/intermediate_data.hpp" + +#include "base/file_name_utils.hpp" + +#include "defines.hpp" + +using namespace feature; + +namespace generator +{ +TranslatorComplex::TranslatorComplex(std::shared_ptr const & processor, + std::shared_ptr const & cache) + : Translator(processor, cache, std::make_shared(cache)) +{ + auto filters = std::make_shared(); + filters->Append(std::make_shared()); + filters->Append(std::make_shared()); + filters->Append(std::make_shared( + base::JoinPath(GetPlatform().ResourcesDir(), SKIPPED_ELEMENTS_FILE))); + SetFilter(filters); +} + +std::shared_ptr TranslatorComplex::Clone() const +{ + return Translator::CloneBase(); +} + +void TranslatorComplex::Merge(TranslatorInterface const & other) { other.MergeInto(*this); } + +void TranslatorComplex::MergeInto(TranslatorComplex & other) const { MergeIntoBase(other); } +} // namespace generator diff --git a/generator/translator_complex.hpp b/generator/translator_complex.hpp new file mode 100644 index 0000000000..d241f6ca49 --- /dev/null +++ b/generator/translator_complex.hpp @@ -0,0 +1,32 @@ +#pragma once + +#include "generator/processor_interface.hpp" +#include "generator/translator.hpp" + +#include +#include + +namespace cache +{ +class IntermediateData; +} // namespace cache + +namespace generator +{ +// The TranslatorComplex class implements translator for building map objects complex. +class TranslatorComplex : public Translator +{ +public: + explicit TranslatorComplex(std::shared_ptr const & processor, + std::shared_ptr const & cache); + + // TranslatorInterface overrides: + std::shared_ptr Clone() const override; + + void Merge(TranslatorInterface const & other) override; + void MergeInto(TranslatorComplex & other) const override; + +protected: + using Translator::Translator; +}; +} // namespace generator diff --git a/generator/translator_factory.hpp b/generator/translator_factory.hpp index 6260f3aefd..d41e583f8f 100644 --- a/generator/translator_factory.hpp +++ b/generator/translator_factory.hpp @@ -2,6 +2,7 @@ #include "generator/factory_utils.hpp" #include "generator/translator_coastline.hpp" +#include "generator/translator_complex.hpp" #include "generator/translator_country.hpp" #include "generator/translator_interface.hpp" #include "generator/translator_world.hpp" @@ -18,6 +19,7 @@ enum class TranslatorType Country, Coastline, World, + Complex }; template @@ -28,6 +30,8 @@ std::shared_ptr CreateTranslator(TranslatorType type, Args case TranslatorType::Coastline: return create(std::forward(args)...); case TranslatorType::Country: return create(std::forward(args)...); case TranslatorType::World: return create(std::forward(args)...); + case TranslatorType::Complex: + return create(std::forward(args)...); } UNREACHABLE(); } diff --git a/generator/translator_interface.hpp b/generator/translator_interface.hpp index 351fc54176..e81260d191 100644 --- a/generator/translator_interface.hpp +++ b/generator/translator_interface.hpp @@ -19,6 +19,7 @@ class TranslatorCountry; class TranslatorCoastline; class TranslatorWorld; class TranslatorCollection; +class TranslatorComplex; // Implementing this interface allows an object to create intermediate data from OsmElement. class TranslatorInterface @@ -39,6 +40,7 @@ public: virtual void MergeInto(TranslatorCoastline &) const { FailIfMethodUnsupported(); } virtual void MergeInto(TranslatorWorld &) const { FailIfMethodUnsupported(); } virtual void MergeInto(TranslatorCollection &) const { FailIfMethodUnsupported(); } + virtual void MergeInto(TranslatorComplex &) const { FailIfMethodUnsupported(); } private: void FailIfMethodUnsupported() const { CHECK(false, ("This method is unsupported.")); } diff --git a/generator/utils.cpp b/generator/utils.cpp index 48d02a9215..eeeea2a148 100644 --- a/generator/utils.cpp +++ b/generator/utils.cpp @@ -12,12 +12,47 @@ #include "base/assert.hpp" #include "base/cancellable.hpp" +#include "base/exception.hpp" #include "base/logging.hpp" +#include +#include #include +#define BOOST_STACKTRACE_GNU_SOURCE_NOT_REQUIRED +#include + namespace generator { +void ErrorHandler(int signum) +{ + // Avoid recursive calls. + std::signal(signum, SIG_DFL); + // If there was an exception, then we will print the message. + try + { + if (auto const eptr = std::current_exception()) + std::rethrow_exception(eptr); + } + catch (RootException const & e) + { + std::cerr << "Core exception: " << e.Msg() << "\n"; + } + catch (std::exception const & e) + { + std::cerr << "Std exception: " << e.what() << "\n"; + } + catch (...) + { + std::cerr << "Unknown exception.\n"; + } + + // Print stack stack. + std::cerr << boost::stacktrace::stacktrace(); + // We raise the signal SIGABRT, so that there would be an opportunity to make a core dump. + std::raise(SIGABRT); +} + // SingleMwmDataSource ----------------------------------------------------------------------------- SingleMwmDataSource::SingleMwmDataSource(std::string const & mwmPath) { @@ -32,6 +67,17 @@ SingleMwmDataSource::SingleMwmDataSource(std::string const & mwmPath) m_mwmId = result.first; } +FeatureGetter::FeatureGetter(std::string const & countryFullPath) + : m_mwm(countryFullPath) + , m_guard(std::make_unique(m_mwm.GetDataSource(), m_mwm.GetMwmId())) +{ +} + +std::unique_ptr FeatureGetter::GetFeatureByIndex(uint32_t index) const +{ + return m_guard->GetFeatureByIndex(index); +} + void LoadDataSource(DataSource & dataSource) { std::vector localFiles; diff --git a/generator/utils.hpp b/generator/utils.hpp index 930a54c50d..5fc4ed9b57 100644 --- a/generator/utils.hpp +++ b/generator/utils.hpp @@ -16,12 +16,24 @@ #include "base/logging.hpp" +#include #include #include #include +#include + +#define MAIN_WITH_ERROR_HANDLING(func) \ + int main(int argc, char ** argv) \ + { \ + std::signal(SIGABRT, generator::ErrorHandler); \ + std::signal(SIGSEGV, generator::ErrorHandler); \ + return func(argc, argv); \ + } namespace generator { +void ErrorHandler(int signum); + /// \brief This class is wrapper around |DataSource| if only one mwm is registered in DataSource. class SingleMwmDataSource { @@ -41,6 +53,18 @@ private: void LoadDataSource(DataSource & dataSource); +class FeatureGetter +{ +public: + FeatureGetter(std::string const & countryFullPath); + + std::unique_ptr GetFeatureByIndex(uint32_t index) const; + +private: + SingleMwmDataSource m_mwm; + std::unique_ptr m_guard; +}; + template bool ForEachOsmId2FeatureId(std::string const & path, ToDo && toDo) { @@ -60,7 +84,6 @@ bool ForEachOsmId2FeatureId(std::string const & path, ToDo && toDo) mapping.ForEach([&](auto const & p) { toDo(p.first /* osm id */, p.second /* feature id */); }); - return true; } diff --git a/indexer/ftypes_matcher.cpp b/indexer/ftypes_matcher.cpp index 8b1b29f7c1..16d24cff88 100644 --- a/indexer/ftypes_matcher.cpp +++ b/indexer/ftypes_matcher.cpp @@ -130,6 +130,11 @@ bool BaseChecker::IsMatched(uint32_t type) const return (find(m_types.begin(), m_types.end(), PrepareToMatch(type, m_level)) != m_types.end()); } +void BaseChecker::ForEachType(function && fn) const +{ + for_each(m_types.cbegin(), m_types.cend(), move(fn)); +} + bool BaseChecker::operator()(feature::TypesHolder const & types) const { for (uint32_t t : types) @@ -478,56 +483,6 @@ unsigned IsHotelChecker::GetHotelTypesMask(FeatureType & ft) const return mask; } -IsPopularityPlaceChecker::IsPopularityPlaceChecker() -{ - vector> const popularityPlaceTypes = { - {"amenity", "bar"}, - {"amenity", "biergarten"}, - {"amenity", "cafe"}, - {"amenity", "casino"}, - {"amenity", "cinema"}, - {"amenity", "fast_food"}, - {"amenity", "fountain"}, - {"amenity", "grave_yard"}, - {"amenity", "marketplace"}, - {"amenity", "nightclub"}, - {"amenity", "place_of_worship"}, - {"amenity", "pub"}, - {"amenity", "restaurant"}, - {"amenity", "theatre"}, - {"amenity", "townhall"}, - {"highway", "pedestrian"}, - {"historic", "archaeological_site"}, - {"historic", "castle"}, - {"historic", "memorial"}, - {"historic", "monument"}, - {"historic", "museum"}, - {"historic", "ruins"}, - {"historic", "ship"}, - {"historic", "tomb"}, - {"landuse", "cemetery"}, - {"leisure", "garden"}, - {"leisure", "park"}, - {"leisure", "water_park"}, - {"man_made", "lighthouse"}, - {"natural", "geyser"}, - {"natural", "peak"}, - {"shop", "bakery"}, - {"tourism", "artwork"}, - {"tourism", "attraction"}, - {"tourism", "gallery"}, - {"tourism", "museum"}, - {"tourism", "theme_park"}, - {"tourism", "viewpoint"}, - {"tourism", "zoo"}, - {"waterway", "waterfall"} - }; - - Classificator const & c = classif(); - for (auto const & t : popularityPlaceTypes) - m_types.push_back(c.GetTypeByPath({t.first, t.second})); -} - IsIslandChecker::IsIslandChecker() { vector> const types = { diff --git a/indexer/ftypes_matcher.hpp b/indexer/ftypes_matcher.hpp index 7ef0ce3d16..c56fcfb098 100644 --- a/indexer/ftypes_matcher.hpp +++ b/indexer/ftypes_matcher.hpp @@ -29,14 +29,15 @@ namespace ftypes class BaseChecker { protected: - size_t const m_level; + uint8_t const m_level; std::vector m_types; - BaseChecker(size_t level = 2) : m_level(level) {} + BaseChecker(uint8_t level = 2) : m_level(level) {} virtual ~BaseChecker() = default; public: virtual bool IsMatched(uint32_t type) const; + virtual void ForEachType(std::function && fn) const; bool operator()(feature::TypesHolder const & types) const; bool operator()(FeatureType & ft) const; @@ -44,12 +45,6 @@ public: bool operator()(uint32_t type) const { return IsMatched(type); } static uint32_t PrepareToMatch(uint32_t type, uint8_t level); - - template - void ForEachType(TFn && fn) const - { - std::for_each(m_types.cbegin(), m_types.cend(), std::forward(fn)); - } }; class IsPeakChecker : public BaseChecker @@ -239,13 +234,6 @@ public: DECLARE_CHECKER_INSTANCE(IsTunnelChecker); }; -class IsPopularityPlaceChecker : public BaseChecker -{ - IsPopularityPlaceChecker(); -public: - DECLARE_CHECKER_INSTANCE(IsPopularityPlaceChecker); -}; - class IsIslandChecker : public BaseChecker { IsIslandChecker();