forked from organicmaps/organicmaps
Match against relations.
This commit is contained in:
parent
36fc19a3d2
commit
d4aa42b8a4
6 changed files with 390 additions and 75 deletions
|
@ -15,6 +15,20 @@
|
|||
|
||||
using editor::XMLFeature;
|
||||
|
||||
namespace
|
||||
{
|
||||
m2::RectD GetBoundingRect(vector<m2::PointD> 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<m2::PointD> 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<m2::PointD> 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.
|
||||
|
|
|
@ -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;
|
||||
|
|
|
@ -5,7 +5,9 @@
|
|||
|
||||
#include "3party/pugixml/src/pugixml.hpp"
|
||||
|
||||
static char const * const osmRawResponseWay = R"SEP(
|
||||
namespace
|
||||
{
|
||||
char const * const osmRawResponseWay = R"SEP(
|
||||
<osm version="0.6" generator="CGImap 0.4.0 (22123 thorn-03.openstreetmap.org)" copyright="OpenStreetMap and contributors">
|
||||
<bounds minlat="53.8976570" minlon="27.5576615" maxlat="53.8976570" maxlon="27.5576615"/>
|
||||
<node id="277171984" visible="true" version="2" changeset="20577443" timestamp="2014-02-15T14:37:39Z" user="iglezz" uid="450366" lat="53.8978034" lon="27.5577642"/>
|
||||
|
@ -52,7 +54,7 @@ static char const * const osmRawResponseWay = R"SEP(
|
|||
</osm>
|
||||
)SEP";
|
||||
|
||||
static char const * const osmRawResponseNode = R"SEP(
|
||||
char const * const osmRawResponseNode = R"SEP(
|
||||
<osm version="0.6" generator="CGImap 0.4.0 (5501 thorn-02.openstreetmap.org)" copyright="OpenStreetMap and contributors">
|
||||
<bounds minlat="53.8977000" minlon="27.5578900" maxlat="53.8977700" maxlon="27.5579800"/>
|
||||
<node id="2673014342" visible="true" version="1" changeset="20577443" timestamp="2014-02-15T14:37:11Z" user="iglezz" uid="450366" lat="53.8976095" lon="27.5579360"/>
|
||||
|
@ -84,41 +86,182 @@ static char const * const osmRawResponseNode = R"SEP(
|
|||
</osm>
|
||||
)SEP";
|
||||
|
||||
UNIT_TEST(GetBestOsmWay_Test)
|
||||
{
|
||||
{
|
||||
pugi::xml_document osmResponse;
|
||||
TEST(osmResponse.load_buffer(osmRawResponseWay, ::strlen(osmRawResponseWay)), ());
|
||||
vector<m2::PointD> 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<m2::PointD> 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(
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<osm version="0.6" generator="CGImap 0.4.0 (22560 thorn-01.openstreetmap.org)" copyright="OpenStreetMap and contributors">
|
||||
<bounds minlat="55.7509200" minlon="37.6397200" maxlat="55.7515400" maxlon="37.6411300"/>
|
||||
<node id="271892032" visible="true" version="4" changeset="8261156" timestamp="2011-05-27T10:18:35Z" user="Scondo" uid="421524" lat="55.7524913" lon="37.6397264"/>
|
||||
<node id="271892033" visible="true" version="4" changeset="8261156" timestamp="2011-05-27T10:18:17Z" user="Scondo" uid="421524" lat="55.7522475" lon="37.6391447"/>
|
||||
<node id="583193392" visible="true" version="2" changeset="4128036" timestamp="2010-03-14T18:31:07Z" user="Vovanium" uid="87682" lat="55.7507909" lon="37.6404902"/>
|
||||
<node id="583193432" visible="true" version="3" changeset="4128036" timestamp="2010-03-14T18:31:08Z" user="Vovanium" uid="87682" lat="55.7510964" lon="37.6397197"/>
|
||||
<node id="583193426" visible="true" version="3" changeset="4128036" timestamp="2010-03-14T18:31:08Z" user="Vovanium" uid="87682" lat="55.7510560" lon="37.6394035"/>
|
||||
<node id="583193429" visible="true" version="3" changeset="4128036" timestamp="2010-03-14T18:31:08Z" user="Vovanium" uid="87682" lat="55.7512865" lon="37.6396919"/>
|
||||
<node id="583193395" visible="true" version="2" changeset="4128036" timestamp="2010-03-14T18:31:08Z" user="Vovanium" uid="87682" lat="55.7509787" lon="37.6401799"/>
|
||||
<node id="583193415" visible="true" version="3" changeset="4128036" timestamp="2010-03-14T18:31:08Z" user="Vovanium" uid="87682" lat="55.7510898" lon="37.6403571"/>
|
||||
<node id="583193424" visible="true" version="3" changeset="4128036" timestamp="2010-03-14T18:31:08Z" user="Vovanium" uid="87682" lat="55.7508581" lon="37.6399029"/>
|
||||
<node id="583193422" visible="true" version="3" changeset="4128036" timestamp="2010-03-14T18:31:08Z" user="Vovanium" uid="87682" lat="55.7509689" lon="37.6400415"/>
|
||||
<node id="583193398" visible="true" version="2" changeset="4128036" timestamp="2010-03-14T18:31:08Z" user="Vovanium" uid="87682" lat="55.7514775" lon="37.6401937"/>
|
||||
<node id="583193416" visible="true" version="3" changeset="4128036" timestamp="2010-03-14T18:31:08Z" user="Vovanium" uid="87682" lat="55.7513532" lon="37.6405069"/>
|
||||
<node id="583193431" visible="true" version="3" changeset="4128036" timestamp="2010-03-14T18:31:08Z" user="Vovanium" uid="87682" lat="55.7512162" lon="37.6398695"/>
|
||||
<node id="583193390" visible="true" version="2" changeset="4128036" timestamp="2010-03-14T18:31:08Z" user="Vovanium" uid="87682" lat="55.7507783" lon="37.6410989"/>
|
||||
<node id="583193388" visible="true" version="2" changeset="4128036" timestamp="2010-03-14T18:31:08Z" user="Vovanium" uid="87682" lat="55.7509982" lon="37.6416194"/>
|
||||
<node id="583193405" visible="true" version="3" changeset="4128036" timestamp="2010-03-14T18:31:08Z" user="Vovanium" uid="87682" lat="55.7514149" lon="37.6406910"/>
|
||||
<node id="583193408" visible="true" version="2" changeset="4128036" timestamp="2010-03-14T18:31:08Z" user="Vovanium" uid="87682" lat="55.7509930" lon="37.6412441"/>
|
||||
<node id="583193410" visible="true" version="3" changeset="4128036" timestamp="2010-03-14T18:31:08Z" user="Vovanium" uid="87682" lat="55.7509124" lon="37.6406648"/>
|
||||
<node id="583193401" visible="true" version="2" changeset="4128036" timestamp="2010-03-14T18:31:09Z" user="Vovanium" uid="87682" lat="55.7516648" lon="37.6407506"/>
|
||||
<node id="666179513" visible="true" version="1" changeset="4128036" timestamp="2010-03-14T18:30:48Z" user="Vovanium" uid="87682" lat="55.7510740" lon="37.6401059"/>
|
||||
<node id="666179517" visible="true" version="1" changeset="4128036" timestamp="2010-03-14T18:30:48Z" user="Vovanium" uid="87682" lat="55.7511507" lon="37.6403206"/>
|
||||
<node id="666179519" visible="true" version="1" changeset="4128036" timestamp="2010-03-14T18:30:48Z" user="Vovanium" uid="87682" lat="55.7512863" lon="37.6403782"/>
|
||||
<node id="666179521" visible="true" version="1" changeset="4128036" timestamp="2010-03-14T18:30:48Z" user="Vovanium" uid="87682" lat="55.7512185" lon="37.6403206"/>
|
||||
<node id="666179522" visible="true" version="1" changeset="4128036" timestamp="2010-03-14T18:30:48Z" user="Vovanium" uid="87682" lat="55.7512214" lon="37.6400483"/>
|
||||
<node id="666179524" visible="true" version="1" changeset="4128036" timestamp="2010-03-14T18:30:48Z" user="Vovanium" uid="87682" lat="55.7513393" lon="37.6401111"/>
|
||||
<node id="666179526" visible="true" version="1" changeset="4128036" timestamp="2010-03-14T18:30:48Z" user="Vovanium" uid="87682" lat="55.7514337" lon="37.6402525"/>
|
||||
<node id="666179528" visible="true" version="1" changeset="4128036" timestamp="2010-03-14T18:30:48Z" user="Vovanium" uid="87682" lat="55.7507439" lon="37.6406349"/>
|
||||
<node id="666179530" visible="true" version="1" changeset="4128036" timestamp="2010-03-14T18:30:48Z" user="Vovanium" uid="87682" lat="55.7507291" lon="37.6407868"/>
|
||||
<node id="666179531" visible="true" version="1" changeset="4128036" timestamp="2010-03-14T18:30:48Z" user="Vovanium" uid="87682" lat="55.7507380" lon="37.6409544"/>
|
||||
<node id="666179540" visible="true" version="1" changeset="4128036" timestamp="2010-03-14T18:30:48Z" user="Vovanium" uid="87682" lat="55.7508736" lon="37.6408287"/>
|
||||
<node id="595699492" visible="true" version="2" changeset="4128036" timestamp="2010-03-14T18:31:08Z" user="Vovanium" uid="87682" lat="55.7508900" lon="37.6410015"/>
|
||||
<node id="271892037" visible="true" version="4" changeset="8261156" timestamp="2011-05-27T10:18:03Z" user="Scondo" uid="421524" lat="55.7519689" lon="37.6393462"/>
|
||||
<node id="666179544" visible="true" version="2" changeset="8261156" timestamp="2011-05-27T10:18:03Z" user="Scondo" uid="421524" lat="55.7523858" lon="37.6394615"/>
|
||||
<node id="271892040" visible="true" version="4" changeset="8261156" timestamp="2011-05-27T10:18:09Z" user="Scondo" uid="421524" lat="55.7518044" lon="37.6401900"/>
|
||||
<node id="271892039" visible="true" version="4" changeset="8261156" timestamp="2011-05-27T10:18:11Z" user="Scondo" uid="421524" lat="55.7518997" lon="37.6400631"/>
|
||||
<node id="271892031" visible="true" version="4" changeset="8261156" timestamp="2011-05-27T10:18:23Z" user="Scondo" uid="421524" lat="55.7517772" lon="37.6406618"/>
|
||||
<node id="271892036" visible="true" version="4" changeset="8261156" timestamp="2011-05-27T10:18:23Z" user="Scondo" uid="421524" lat="55.7521424" lon="37.6397730"/>
|
||||
<node id="271892035" visible="true" version="4" changeset="8261156" timestamp="2011-05-27T10:18:25Z" user="Scondo" uid="421524" lat="55.7522520" lon="37.6396264"/>
|
||||
<node id="666179542" visible="true" version="2" changeset="8261156" timestamp="2011-05-27T10:18:26Z" user="Scondo" uid="421524" lat="55.7523415" lon="37.6393631"/>
|
||||
<node id="271892038" visible="true" version="4" changeset="8261156" timestamp="2011-05-27T10:18:30Z" user="Scondo" uid="421524" lat="55.7517353" lon="37.6396389"/>
|
||||
<node id="666179545" visible="true" version="2" changeset="8261156" timestamp="2011-05-27T10:18:30Z" user="Scondo" uid="421524" lat="55.7523947" lon="37.6392844"/>
|
||||
<node id="271892041" visible="true" version="4" changeset="8261156" timestamp="2011-05-27T10:18:34Z" user="Scondo" uid="421524" lat="55.7516804" lon="37.6398672"/>
|
||||
<node id="666179548" visible="true" version="2" changeset="8261156" timestamp="2011-05-27T10:18:35Z" user="Scondo" uid="421524" lat="55.7524390" lon="37.6393828"/>
|
||||
<node id="271892030" visible="true" version="4" changeset="8261156" timestamp="2011-05-27T10:18:38Z" user="Scondo" uid="421524" lat="55.7515240" lon="37.6400640"/>
|
||||
<node id="271892034" visible="true" version="4" changeset="8261156" timestamp="2011-05-27T10:18:40Z" user="Scondo" uid="421524" lat="55.7521203" lon="37.6393028"/>
|
||||
<node id="2849850611" visible="true" version="2" changeset="33550372" timestamp="2015-08-24T15:55:36Z" user="vadp" uid="326091" lat="55.7507261" lon="37.6405934"/>
|
||||
<node id="2849850614" visible="true" version="1" changeset="22264538" timestamp="2014-05-11T07:26:43Z" user="Vadim Zudkin" uid="177747" lat="55.7509233" lon="37.6401297"/>
|
||||
<node id="3712207029" visible="true" version="1" changeset="33550372" timestamp="2015-08-24T15:55:35Z" user="vadp" uid="326091" lat="55.7510865" lon="37.6400013"/>
|
||||
<node id="3712207030" visible="true" version="1" changeset="33550372" timestamp="2015-08-24T15:55:35Z" user="vadp" uid="326091" lat="55.7512462" lon="37.6399456"/>
|
||||
<node id="3712207031" visible="true" version="2" changeset="33550412" timestamp="2015-08-24T15:57:10Z" user="vadp" uid="326091" lat="55.7514944" lon="37.6401534"/>
|
||||
<node id="3712207032" visible="true" version="2" changeset="33550412" timestamp="2015-08-24T15:57:10Z" user="vadp" uid="326091" lat="55.7516969" lon="37.6407362">
|
||||
<tag k="access" v="private"/>
|
||||
<tag k="barrier" v="gate"/>
|
||||
</node>
|
||||
<node id="3712207033" visible="true" version="2" changeset="33550412" timestamp="2015-08-24T15:57:10Z" user="vadp" uid="326091" lat="55.7517316" lon="37.6408217"/>
|
||||
<node id="3712207034" visible="true" version="2" changeset="33550412" timestamp="2015-08-24T15:57:10Z" user="vadp" uid="326091" lat="55.7517602" lon="37.6409066"/>
|
||||
<node id="2849850613" visible="true" version="3" changeset="33551686" timestamp="2015-08-24T16:50:21Z" user="vadp" uid="326091" lat="55.7507965" lon="37.6399611"/>
|
||||
<node id="338464706" visible="true" version="3" changeset="33551686" timestamp="2015-08-24T16:50:21Z" user="vadp" uid="326091" lat="55.7510322" lon="37.6393637"/>
|
||||
<node id="338464708" visible="true" version="7" changeset="33551686" timestamp="2015-08-24T16:50:21Z" user="vadp" uid="326091" lat="55.7515407" lon="37.6383137"/>
|
||||
<node id="3755931947" visible="true" version="1" changeset="34206452" timestamp="2015-09-23T13:58:11Z" user="trolleway" uid="397326" lat="55.7517090" lon="37.6407565"/>
|
||||
<way id="25009838" visible="true" version="14" changeset="28090002" timestamp="2015-01-12T16:15:21Z" user="midrug" uid="2417727">
|
||||
<nd ref="271892030"/>
|
||||
<nd ref="271892031"/>
|
||||
<nd ref="271892032"/>
|
||||
<nd ref="666179544"/>
|
||||
<nd ref="666179548"/>
|
||||
<nd ref="666179545"/>
|
||||
<nd ref="666179542"/>
|
||||
<nd ref="271892033"/>
|
||||
<nd ref="271892034"/>
|
||||
<nd ref="271892035"/>
|
||||
<nd ref="271892036"/>
|
||||
<nd ref="271892037"/>
|
||||
<nd ref="271892038"/>
|
||||
<nd ref="271892039"/>
|
||||
<nd ref="271892040"/>
|
||||
<nd ref="271892041"/>
|
||||
<nd ref="271892030"/>
|
||||
<tag k="addr:housenumber" v="12-14"/>
|
||||
<tag k="addr:street" v="улица Солянка"/>
|
||||
<tag k="building" v="yes"/>
|
||||
<tag k="building:colour" v="lightpink"/>
|
||||
<tag k="building:levels" v="3"/>
|
||||
<tag k="description:en" v="Housed the Board of Trustees, a public institution of the Russian Empire until 1917"/>
|
||||
<tag k="end_date" v="1826"/>
|
||||
<tag k="name" v="Опекунский совет"/>
|
||||
<tag k="name:de" v="Kuratorium"/>
|
||||
<tag k="name:en" v="Board of Trustees Building"/>
|
||||
<tag k="ref" v="7710784000"/>
|
||||
<tag k="roof:material" v="metal"/>
|
||||
<tag k="source:description:en" v="wikipedia:ru"/>
|
||||
<tag k="start_date" v="1823"/>
|
||||
<tag k="tourism" v="attraction"/>
|
||||
<tag k="wikipedia" v="ru:Опекунский совет (Москва)"/>
|
||||
</way>
|
||||
<way id="45814282" visible="true" version="3" changeset="4128036" timestamp="2010-03-14T18:31:24Z" user="Vovanium" uid="87682">
|
||||
<nd ref="583193405"/>
|
||||
<nd ref="583193408"/>
|
||||
<nd ref="595699492"/>
|
||||
<nd ref="666179540"/>
|
||||
<nd ref="583193410"/>
|
||||
<nd ref="583193415"/>
|
||||
<nd ref="666179517"/>
|
||||
<nd ref="666179521"/>
|
||||
<nd ref="666179519"/>
|
||||
<nd ref="583193416"/>
|
||||
<nd ref="583193405"/>
|
||||
</way>
|
||||
<way id="109538181" visible="true" version="7" changeset="34206452" timestamp="2015-09-23T13:58:11Z" user="trolleway" uid="397326">
|
||||
<nd ref="338464708"/>
|
||||
<nd ref="338464706"/>
|
||||
<nd ref="2849850613"/>
|
||||
<nd ref="2849850614"/>
|
||||
<nd ref="3712207029"/>
|
||||
<nd ref="3712207030"/>
|
||||
<nd ref="3712207031"/>
|
||||
<nd ref="3712207032"/>
|
||||
<nd ref="3755931947"/>
|
||||
<nd ref="3712207033"/>
|
||||
<nd ref="3712207034"/>
|
||||
<tag k="highway" v="service"/>
|
||||
</way>
|
||||
<way id="45814281" visible="true" version="6" changeset="9583527" timestamp="2011-10-17T16:38:55Z" user="luch86" uid="266092">
|
||||
<nd ref="583193388"/>
|
||||
<nd ref="583193390"/>
|
||||
<nd ref="666179531"/>
|
||||
<nd ref="666179530"/>
|
||||
<nd ref="666179528"/>
|
||||
<nd ref="583193392"/>
|
||||
<nd ref="583193395"/>
|
||||
<nd ref="666179513"/>
|
||||
<nd ref="666179522"/>
|
||||
<nd ref="666179524"/>
|
||||
<nd ref="666179526"/>
|
||||
<nd ref="583193398"/>
|
||||
<nd ref="583193401"/>
|
||||
<nd ref="583193388"/>
|
||||
</way>
|
||||
<way id="45814283" visible="true" version="3" changeset="28090002" timestamp="2015-01-12T16:15:23Z" user="midrug" uid="2417727">
|
||||
<nd ref="583193422"/>
|
||||
<nd ref="583193424"/>
|
||||
<nd ref="583193426"/>
|
||||
<nd ref="583193429"/>
|
||||
<nd ref="583193431"/>
|
||||
<nd ref="583193432"/>
|
||||
<nd ref="583193422"/>
|
||||
<tag k="addr:street" v="улица Солянка"/>
|
||||
<tag k="building" v="yes"/>
|
||||
<tag k="building:colour" v="goldenrod"/>
|
||||
<tag k="building:levels" v="2"/>
|
||||
<tag k="roof:colour" v="black"/>
|
||||
<tag k="roof:material" v="tar_paper"/>
|
||||
</way>
|
||||
<way id="367274913" visible="true" version="2" changeset="33550484" timestamp="2015-08-24T16:00:25Z" user="vadp" uid="326091">
|
||||
<nd ref="2849850614"/>
|
||||
<nd ref="2849850611"/>
|
||||
<tag k="highway" v="service"/>
|
||||
</way>
|
||||
<relation id="365808" visible="true" version="6" changeset="28090002" timestamp="2015-01-12T16:15:14Z" user="midrug" uid="2417727">
|
||||
<member type="way" ref="45814281" role="outer"/>
|
||||
<member type="way" ref="45814282" role="inner"/>
|
||||
<tag k="addr:housenumber" v="14/2"/>
|
||||
<tag k="addr:street" v="улица Солянка"/>
|
||||
<tag k="building" v="yes"/>
|
||||
<tag k="building:colour" v="gold"/>
|
||||
<tag k="building:levels" v="2"/>
|
||||
<tag k="roof:material" v="metal"/>
|
||||
<tag k="type" v="multipolygon"/>
|
||||
</relation>
|
||||
</osm>
|
||||
)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<m2::PointD> 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<m2::PointD> 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<m2::PointD> 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"), ());
|
||||
}
|
||||
|
|
|
@ -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 <typename TFunc>
|
||||
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<TFunc>(func));
|
||||
}
|
||||
}
|
||||
|
||||
vector<m2::PointD> GetWaysGeometry(pugi::xml_document const & osmResponse,
|
||||
pugi::xml_node const & way)
|
||||
{
|
||||
|
@ -49,54 +68,77 @@ vector<m2::PointD> GetWaysGeometry(pugi::xml_document const & osmResponse,
|
|||
return result;
|
||||
}
|
||||
|
||||
vector<m2::PointD> GetRelationsGeometry(pugi::xml_document const & osmResponse,
|
||||
pugi::xml_node const & relation)
|
||||
{
|
||||
vector<m2::PointD> result;
|
||||
ForEachRelationsNode(osmResponse, relation, [&result](XMLFeature const & xmlFt)
|
||||
{
|
||||
result.push_back(xmlFt.GetMercatorCenter());
|
||||
});
|
||||
return result;
|
||||
}
|
||||
|
||||
// TODO(mgsergio): XMLFeature should have GetGeometry method.
|
||||
vector<m2::PointD> 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<m2::PointD> 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<m2::PointD> 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<double>(matched) / wayGeometry.size() - 0.5;
|
||||
auto const geomScore = static_cast<double>(matched) / geometry.size() - 0.5;
|
||||
auto const wayScore = static_cast<double>(matched) / theirGeometry.size() - 0.5;
|
||||
auto const geomScore = static_cast<double>(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<m2::PointD> const & geometry)
|
||||
pugi::xml_node GetBestOsmWayOrRelation(pugi::xml_document const & osmResponse,
|
||||
vector<m2::PointD> 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();
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -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<m2::PointD> const & geometry);
|
||||
pugi::xml_node GetBestOsmWayOrRelation(pugi::xml_document const & osmResponse,
|
||||
vector<m2::PointD> const & geometry);
|
||||
} // namespace osm
|
||||
|
|
|
@ -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();
|
||||
|
|
Loading…
Add table
Reference in a new issue