diff --git a/editor/changeset_wrapper.cpp b/editor/changeset_wrapper.cpp index 3a99160fb5..e844e23218 100644 --- a/editor/changeset_wrapper.cpp +++ b/editor/changeset_wrapper.cpp @@ -15,6 +15,20 @@ using editor::XMLFeature; +namespace +{ +m2::RectD GetBoundingRect(vector const & geometry) +{ + m2::RectD rect; + for (auto const & p : geometry) + { + auto const latLon = MercatorBounds::ToLatLon(p); + rect.Add({latLon.lon, latLon.lat}); + } + return rect; +} +} // namespace + namespace pugi { string DebugPrint(xml_document const & doc) @@ -27,7 +41,6 @@ string DebugPrint(xml_document const & doc) namespace osm { - ChangesetWrapper::ChangesetWrapper(TKeySecret const & keySecret, ServerApi06::TKeyValueTags const & comments) noexcept : m_changesetComments(comments), m_api(OsmOAuth::ServerAuth(keySecret)) @@ -59,6 +72,16 @@ void ChangesetWrapper::LoadXmlFromOSM(ms::LatLon const & ll, pugi::xml_document MYTHROW(OsmXmlParseException, ("Can't parse OSM server response for GetXmlFeaturesAtLatLon request", response.second)); } +void ChangesetWrapper::LoadXmlFromOSM(m2::RectD const & rect, pugi::xml_document & doc) +{ + auto const response = m_api.GetXmlFeaturesInRect(rect); + if (response.first != OsmOAuth::HTTP::OK) + MYTHROW(HttpErrorException, ("HTTP error", response, "with GetXmlFeaturesInRect", rect)); + + if (pugi::status_ok != doc.load(response.second.c_str()).status) + MYTHROW(OsmXmlParseException, ("Can't parse OSM server response for GetXmlFeaturesInRect request", response.second)); +} + XMLFeature ChangesetWrapper::GetMatchingNodeFeatureFromOSM(m2::PointD const & center) { // Match with OSM node. @@ -80,6 +103,7 @@ XMLFeature ChangesetWrapper::GetMatchingNodeFeatureFromOSM(m2::PointD const & ce XMLFeature ChangesetWrapper::GetMatchingAreaFeatureFromOSM(vector const & geometry) { // TODO: Make two/four requests using points on inscribed rectagle. + bool hasRelation = false; for (auto const & pt : geometry) { ms::LatLon const ll = MercatorBounds::ToLatLon(pt); @@ -87,11 +111,32 @@ XMLFeature ChangesetWrapper::GetMatchingAreaFeatureFromOSM(vector co // Throws! LoadXmlFromOSM(ll, doc); - pugi::xml_node const bestWay = GetBestOsmWay(doc, geometry); - if (bestWay.empty()) - continue; + if (doc.select_node("osm/relation")) + { + auto const rect = GetBoundingRect(geometry); + LoadXmlFromOSM(rect, doc); + hasRelation = true; + } - XMLFeature const way(bestWay); + pugi::xml_node const bestWayOrRelation = GetBestOsmWayOrRelation(doc, geometry); + if (!bestWayOrRelation) + { + if (hasRelation) + break; + continue; + } + + if (strcmp(bestWayOrRelation.name(), "relation") == 0) + { + stringstream sstr; + bestWayOrRelation.print(sstr); + LOG(LDEBUG, ("Relation is the best match", sstr.str())); + MYTHROW(RelationFeatureAreNotSupportedException, + ("Got relation as the best matching.")); + } + + // TODO: rename to wayOrRelation when relations are handled. + XMLFeature const way(bestWayOrRelation); ASSERT(way.IsArea(), ("Best way must be an area.")); // AlexZ: TODO: Check that this way is really match our feature. diff --git a/editor/changeset_wrapper.hpp b/editor/changeset_wrapper.hpp index d1665276f5..0eefcf4788 100644 --- a/editor/changeset_wrapper.hpp +++ b/editor/changeset_wrapper.hpp @@ -6,6 +6,7 @@ #include "editor/xml_feature.hpp" #include "geometry/point2d.hpp" +#include "geometry/rect2d.hpp" #include "std/set.hpp" #include "std/vector.hpp" @@ -14,7 +15,6 @@ class FeatureType; namespace osm { - struct ClientToken; class ChangesetWrapper @@ -28,6 +28,8 @@ public: DECLARE_EXCEPTION(CreateChangeSetFailedException, ChangesetWrapperException); DECLARE_EXCEPTION(ModifyNodeFailedException, ChangesetWrapperException); DECLARE_EXCEPTION(LinearFeaturesAreNotSupportedException, ChangesetWrapperException); + // TODO: Remove this when relations are handled properly. + DECLARE_EXCEPTION(RelationFeatureAreNotSupportedException, ChangesetWrapperException); ChangesetWrapper(TKeySecret const & keySecret, ServerApi06::TKeyValueTags const & comments) noexcept; ~ChangesetWrapper(); @@ -51,6 +53,7 @@ private: /// Unfortunately, pugi can't return xml_documents from methods. /// Throws exceptions from above list. void LoadXmlFromOSM(ms::LatLon const & ll, pugi::xml_document & doc); + void LoadXmlFromOSM(m2::RectD const & rect, pugi::xml_document & doc); ServerApi06::TKeyValueTags m_changesetComments; ServerApi06 m_api; diff --git a/editor/editor_tests/osm_feature_matcher_test.cpp b/editor/editor_tests/osm_feature_matcher_test.cpp index f6ac0c3e47..d0a8361b6c 100644 --- a/editor/editor_tests/osm_feature_matcher_test.cpp +++ b/editor/editor_tests/osm_feature_matcher_test.cpp @@ -5,7 +5,9 @@ #include "3party/pugixml/src/pugixml.hpp" -static char const * const osmRawResponseWay = R"SEP( +namespace +{ +char const * const osmRawResponseWay = R"SEP( @@ -52,7 +54,7 @@ static char const * const osmRawResponseWay = R"SEP( )SEP"; -static char const * const osmRawResponseNode = R"SEP( +char const * const osmRawResponseNode = R"SEP( @@ -84,41 +86,182 @@ static char const * const osmRawResponseNode = R"SEP( )SEP"; -UNIT_TEST(GetBestOsmWay_Test) -{ - { - pugi::xml_document osmResponse; - TEST(osmResponse.load_buffer(osmRawResponseWay, ::strlen(osmRawResponseWay)), ()); - vector const geometry = { - MercatorBounds::FromLatLon(53.8977484, 27.557359), - MercatorBounds::FromLatLon(53.8978710, 27.557681), - MercatorBounds::FromLatLon(53.8978034, 27.557764), - MercatorBounds::FromLatLon(53.8977652, 27.557803), - MercatorBounds::FromLatLon(53.8977254, 27.557837), - MercatorBounds::FromLatLon(53.8976570, 27.557661), - MercatorBounds::FromLatLon(53.8976041, 27.557518), - }; - - auto const bestWay = osm::GetBestOsmWay(osmResponse, geometry); - TEST_EQUAL(editor::XMLFeature(bestWay).GetName(), "Беллесбумпром", ()); - } - { - pugi::xml_document osmResponse; - TEST(osmResponse.load_buffer(osmRawResponseWay, ::strlen(osmRawResponseWay)), ()); - vector const geometry = { - MercatorBounds::FromLatLon(53.8975484, 27.557359), // diff - MercatorBounds::FromLatLon(53.8978710, 27.557681), - MercatorBounds::FromLatLon(53.8975034, 27.557764), // diff - MercatorBounds::FromLatLon(53.8977652, 27.557803), - MercatorBounds::FromLatLon(53.8975254, 27.557837), // diff - MercatorBounds::FromLatLon(53.8976570, 27.557661), - MercatorBounds::FromLatLon(53.8976041, 27.557318), // diff - }; - - auto const bestWay = osm::GetBestOsmWay(osmResponse, geometry); - TEST(!bestWay, ()); - } -} +char const * const osmRawResponseRelation = R"SEP( + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +)SEP"; +} // namespace UNIT_TEST(GetBestOsmNode_Test) { @@ -138,3 +281,74 @@ UNIT_TEST(GetBestOsmNode_Test) TEST(bestNode, ()); } } + +UNIT_TEST(GetBestOsmWay_Test) +{ + { + pugi::xml_document osmResponse; + TEST(osmResponse.load_buffer(osmRawResponseWay, ::strlen(osmRawResponseWay)), ()); + vector const geometry = { + MercatorBounds::FromLatLon(53.8977484, 27.557359), + MercatorBounds::FromLatLon(53.8978710, 27.557681), + MercatorBounds::FromLatLon(53.8978034, 27.557764), + MercatorBounds::FromLatLon(53.8977652, 27.557803), + MercatorBounds::FromLatLon(53.8977254, 27.557837), + MercatorBounds::FromLatLon(53.8976570, 27.557661), + MercatorBounds::FromLatLon(53.8976041, 27.557518), + }; + + auto const bestWay = osm::GetBestOsmWayOrRelation(osmResponse, geometry); + TEST_EQUAL(editor::XMLFeature(bestWay).GetName(), "Беллесбумпром", ()); + } + { + pugi::xml_document osmResponse; + TEST(osmResponse.load_buffer(osmRawResponseWay, ::strlen(osmRawResponseWay)), ()); + vector const geometry = { + MercatorBounds::FromLatLon(53.8975484, 27.557359), // diff + MercatorBounds::FromLatLon(53.8978710, 27.557681), + MercatorBounds::FromLatLon(53.8975034, 27.557764), // diff + MercatorBounds::FromLatLon(53.8977652, 27.557803), + MercatorBounds::FromLatLon(53.8975254, 27.557837), // diff + MercatorBounds::FromLatLon(53.8976570, 27.557661), + MercatorBounds::FromLatLon(53.8976041, 27.557318), // diff + }; + + auto const bestWay = osm::GetBestOsmWayOrRelation(osmResponse, geometry); + TEST(!bestWay, ()); + } +} + +UNIT_TEST(GetBestOsmRealtion_Test) +{ + pugi::xml_document osmResponse; + TEST(osmResponse.load_buffer(osmRawResponseRelation, ::strlen(osmRawResponseRelation)), ()); + vector const geometry = { + {37.6400469, 67.4549381}, + {37.6401059, 67.4546779}, + {37.6401113, 67.4551473}, + {37.640181, 67.4545089}, + {37.6401944, 67.455394}, + {37.6402534, 67.4553162}, + {37.6403205, 67.454812}, + {37.6403205, 67.4549327}, + {37.640358, 67.4547047}, + {37.6403768, 67.4550534}, + {37.6404895, 67.4541736}, + {37.6405056, 67.4551741}, + {37.6406343, 67.4540905}, + {37.6406638, 67.4543909}, + {37.6406906, 67.4552814}, + {37.6407496, 67.4557266}, + {37.6407496, 67.45572}, + {37.6407872, 67.4540636}, + {37.6408274, 67.4543211}, + {37.6409535, 67.4540797}, + {37.6410018, 67.4543506}, + {37.6410983, 67.4541522}, + {37.6412432, 67.454533}, + {37.6416187, 67.4545411} + }; + + auto const bestWay = osm::GetBestOsmWayOrRelation(osmResponse, geometry); + TEST_EQUAL(bestWay.attribute("id").value(), string("365808"), ()); +} diff --git a/editor/osm_feature_matcher.cpp b/editor/osm_feature_matcher.cpp index 1605a85729..fac7dc7305 100644 --- a/editor/osm_feature_matcher.cpp +++ b/editor/osm_feature_matcher.cpp @@ -1,13 +1,16 @@ #include "editor/osm_feature_matcher.hpp" #include "base/logging.hpp" +#include "base/stl_helpers.hpp" #include "std/algorithm.hpp" using editor::XMLFeature; -namespace +namespace osm { +using editor::XMLFeature; + constexpr double kPointDiffEps = MercatorBounds::GetCellID2PointAbsEpsilon(); bool PointsEqual(m2::PointD const & a, m2::PointD const & b) @@ -38,6 +41,22 @@ void ForEachWaysNode(pugi::xml_document const & osmResponse, pugi::xml_node cons } } +template +void ForEachRelationsNode(pugi::xml_document const & osmResponse, pugi::xml_node const & relation, + TFunc && func) +{ + for (auto const xNodeRef : relation.select_nodes("member[@type='way']/@ref")) + { + string const wayRef = xNodeRef.attribute().value(); + auto const xpath = "osm/way[@id='" + wayRef + "']"; + auto const way = osmResponse.select_node(xpath.data()).node(); + // Some ways can be missed from relation. + if (!way) + continue; + ForEachWaysNode(osmResponse, way, forward(func)); + } +} + vector GetWaysGeometry(pugi::xml_document const & osmResponse, pugi::xml_node const & way) { @@ -49,54 +68,77 @@ vector GetWaysGeometry(pugi::xml_document const & osmResponse, return result; } +vector GetRelationsGeometry(pugi::xml_document const & osmResponse, + pugi::xml_node const & relation) +{ + vector result; + ForEachRelationsNode(osmResponse, relation, [&result](XMLFeature const & xmlFt) + { + result.push_back(xmlFt.GetMercatorCenter()); + }); + return result; +} + +// TODO(mgsergio): XMLFeature should have GetGeometry method. +vector GetWaysOrRelationsGeometry(pugi::xml_document const & osmResponse, + pugi::xml_node const & wayOrRelation) +{ + if (strcmp(wayOrRelation.name(), "way") == 0) + return GetWaysGeometry(osmResponse, wayOrRelation); + return GetRelationsGeometry(osmResponse, wayOrRelation); +} + /// @returns value form [-0.5, 0.5]. Negative values are used as penalty, /// positive as score. -double ScoreGeometry(pugi::xml_document const & osmResponse, pugi::xml_node const & way, - vector geometry) +/// @param osmResponse - nodes, ways and relations from osm +/// @param wayOrRelation - either way or relation to be compared agains ourGeometry +/// @param outGeometry - geometry of a FeatureType (ourGeometry must be sort-uniqued) +double ScoreGeometry(pugi::xml_document const & osmResponse, + pugi::xml_node const & wayOrRelation, vector ourGeometry) { + ASSERT(!ourGeometry.empty(), ("Our geometry cannot be empty")); int matched = 0; - auto wayGeometry = GetWaysGeometry(osmResponse, way); + auto theirGeometry = GetWaysOrRelationsGeometry(osmResponse, wayOrRelation); - sort(begin(wayGeometry), end(wayGeometry)); - sort(begin(geometry), end(geometry)); + if (theirGeometry.empty()) + return -1; - auto it1 = begin(geometry); - auto it2 = begin(wayGeometry); + my::SortUnique(theirGeometry); - while (it1 != end(geometry) && it2 != end(wayGeometry)) + auto ourIt = begin(ourGeometry); + auto theirIt = begin(theirGeometry); + + while (ourIt != end(ourGeometry) && theirIt != end(theirGeometry)) { - if (PointsEqual(*it1, *it2)) + if (PointsEqual(*ourIt, *theirIt)) { ++matched; - ++it1; - ++it2; + ++ourIt; + ++theirIt; } - else if (*it1 < *it2) + else if (*ourIt < *theirIt) { - ++it1; + ++ourIt; } else { - ++it2; + ++theirIt; } } - auto const wayScore = static_cast(matched) / wayGeometry.size() - 0.5; - auto const geomScore = static_cast(matched) / geometry.size() - 0.5; + auto const wayScore = static_cast(matched) / theirGeometry.size() - 0.5; + auto const geomScore = static_cast(matched) / ourGeometry.size() - 0.5; auto const result = wayScore <= 0 || geomScore <= 0 ? -1 : 2 / (1 / wayScore + 1 / geomScore); - LOG(LDEBUG, ("Osm score:", wayScore, "our feature score:", geomScore, - "Total score", result)); + LOG(LDEBUG, ("Type:", wayOrRelation.name(), "Osm score:", + wayScore, "our feature score:", geomScore, "Total score", result)); return result; } -} // namespace -namespace osm -{ pugi::xml_node GetBestOsmNode(pugi::xml_document const & osmResponse, ms::LatLon const & latLon) { double bestScore = -1; @@ -132,23 +174,23 @@ pugi::xml_node GetBestOsmNode(pugi::xml_document const & osmResponse, ms::LatLon return bestMatchNode; } -pugi::xml_node GetBestOsmWay(pugi::xml_document const & osmResponse, - vector const & geometry) +pugi::xml_node GetBestOsmWayOrRelation(pugi::xml_document const & osmResponse, + vector const & geometry) { double bestScore = -1; pugi::xml_node bestMatchWay; - // TODO(mgsergio): Handle relations as well. Put try_later=version status to edits.xml. - for (auto const & xWay : osmResponse.select_nodes("osm/way")) + auto const xpath = "osm/way|osm/relation[tag[@k='type' and @v='multipolygon']]"; + for (auto const & xWayOrRelation : osmResponse.select_nodes(xpath)) { - double const nodeScore = ScoreGeometry(osmResponse, xWay.node(), geometry); + double const nodeScore = ScoreGeometry(osmResponse, xWayOrRelation.node(), geometry); if (nodeScore < 0) continue; if (bestScore < nodeScore) { bestScore = nodeScore; - bestMatchWay = xWay.node(); + bestMatchWay = xWayOrRelation.node(); } } diff --git a/editor/osm_feature_matcher.hpp b/editor/osm_feature_matcher.hpp index 0c1f05e7bb..2d3947317f 100644 --- a/editor/osm_feature_matcher.hpp +++ b/editor/osm_feature_matcher.hpp @@ -12,6 +12,6 @@ namespace osm /// @returns closest to the latLon node from osm or empty node if none is close enough. pugi::xml_node GetBestOsmNode(pugi::xml_document const & osmResponse, ms::LatLon const & latLon); /// @returns a way from osm with similar geometry or empy node if can't find such way. -pugi::xml_node GetBestOsmWay(pugi::xml_document const & osmResponse, - vector const & geometry); +pugi::xml_node GetBestOsmWayOrRelation(pugi::xml_document const & osmResponse, + vector const & geometry); } // namespace osm diff --git a/indexer/osm_editor.cpp b/indexer/osm_editor.cpp index de1bcd7a54..efb073862c 100644 --- a/indexer/osm_editor.cpp +++ b/indexer/osm_editor.cpp @@ -58,11 +58,15 @@ constexpr char const * kAddrStreetTag = "addr:street"; constexpr char const * kUploaded = "Uploaded"; constexpr char const * kDeletedFromOSMServer = "Deleted from OSM by someone"; +constexpr char const * kRelationsAreNotSupported = "Relations are not supported yet"; constexpr char const * kNeedsRetry = "Needs Retry"; bool NeedsUpload(string const & uploadStatus) { - return uploadStatus != kUploaded && uploadStatus != kDeletedFromOSMServer; + return uploadStatus != kUploaded && + uploadStatus != kDeletedFromOSMServer && + // TODO: Remove this line when relations are supported. + uploadStatus != kRelationsAreNotSupported; } string GetEditorFilePath() { return GetPlatform().WritablePathForFile(kEditorXMLFileName); } @@ -644,9 +648,16 @@ void Editor::UploadChanges(string const & key, string const & secret, TChangeset ++errorsCount; LOG(LWARNING, (ex.what())); } + catch (ChangesetWrapper::RelationFeatureAreNotSupportedException const & ex) + { + fti.m_uploadStatus = kRelationsAreNotSupported; + fti.m_uploadAttemptTimestamp = time(nullptr); + fti.m_uploadError = ex.what(); + ++errorsCount; + LOG(LWARNING, (ex.what())); + } catch (RootException const & ex) { - LOG(LWARNING, (ex.what())); fti.m_uploadStatus = kNeedsRetry; fti.m_uploadAttemptTimestamp = time(nullptr); fti.m_uploadError = ex.what();