[editor] Changeset comments

This commit is contained in:
Ilya Zverev 2016-03-23 18:45:03 +03:00
parent b9c0ef2fd4
commit 14274b5543
4 changed files with 133 additions and 19 deletions

View file

@ -29,9 +29,34 @@ m2::RectD GetBoundingRect(vector<m2::PointD> const & geometry)
return rect;
}
bool OsmFeatureHasTags(pugi::xml_node const & osmFt)
bool OsmFeatureHasTags(pugi::xml_node const & osmFt) { return osmFt.child("tag"); }
vector<string> const static kMainTags = {"amenity", "shop", "tourism", "historic",
"craft", "emergency", "barrier", "highway",
"office", "entrance", "building"};
string GetTypeForFeature(XMLFeature const & node)
{
return osmFt.child("tag");
for (string const & key : kMainTags)
{
if (node.HasTag(key))
{
string value = node.GetTagValue(key);
if (value == "yes")
return key;
else if (key == "shop" || key == "office" || key == "building" || key == "entrance")
return value + " " + key; // "convenience shop"
else
return value;
}
}
// Did not find any known tags.
bool foundTags = false;
node.ForEachTag([&foundTags](string const & k, string const & v)
{
foundTags = true;
});
return foundTags ? "unknown object" : "empty object";
}
vector<m2::PointD> NaiveSample(vector<m2::PointD> const & source, size_t count)
@ -74,7 +99,8 @@ namespace osm
{
ChangesetWrapper::ChangesetWrapper(TKeySecret const & keySecret,
ServerApi06::TKeyValueTags const & comments) noexcept
: m_changesetComments(comments), m_api(OsmOAuth::ServerAuth(keySecret))
: m_changesetComments(comments),
m_api(OsmOAuth::ServerAuth(keySecret))
{
}
@ -84,6 +110,8 @@ ChangesetWrapper::~ChangesetWrapper()
{
try
{
m_changesetComments["comment"] = GetDescription();
m_api.UpdateChangeSet(m_changesetId, m_changesetComments);
m_api.CloseChangeSet(m_changesetId);
}
catch (std::exception const & ex)
@ -100,7 +128,9 @@ void ChangesetWrapper::LoadXmlFromOSM(ms::LatLon const & ll, pugi::xml_document
MYTHROW(HttpErrorException, ("HTTP error", response, "with GetXmlFeaturesAtLatLon", ll));
if (pugi::status_ok != doc.load(response.second.c_str()).status)
MYTHROW(OsmXmlParseException, ("Can't parse OSM server response for GetXmlFeaturesAtLatLon request", response.second));
MYTHROW(
OsmXmlParseException,
("Can't parse OSM server response for GetXmlFeaturesAtLatLon request", response.second));
}
void ChangesetWrapper::LoadXmlFromOSM(m2::RectD const & rect, pugi::xml_document & doc)
@ -110,7 +140,8 @@ void ChangesetWrapper::LoadXmlFromOSM(m2::RectD const & rect, pugi::xml_document
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));
MYTHROW(OsmXmlParseException,
("Can't parse OSM server response for GetXmlFeaturesInRect request", response.second));
}
XMLFeature ChangesetWrapper::GetMatchingNodeFeatureFromOSM(m2::PointD const & center)
@ -171,8 +202,7 @@ XMLFeature ChangesetWrapper::GetMatchingAreaFeatureFromOSM(vector<m2::PointD> co
stringstream sstr;
bestWayOrRelation.print(sstr);
LOG(LDEBUG, ("Relation is the best match", sstr.str()));
MYTHROW(RelationFeatureAreNotSupportedException,
("Got relation as the best matching"));
MYTHROW(RelationFeatureAreNotSupportedException, ("Got relation as the best matching"));
}
if (!OsmFeatureHasTags(bestWayOrRelation))
@ -204,6 +234,7 @@ void ChangesetWrapper::Create(XMLFeature node)
node.SetAttribute("changeset", strings::to_string(m_changesetId));
// TODO(AlexZ): Think about storing/logging returned OSM ids.
UNUSED_VALUE(m_api.CreateElement(node));
m_created_types[GetTypeForFeature(node)]++;
}
void ChangesetWrapper::Modify(XMLFeature node)
@ -214,6 +245,7 @@ void ChangesetWrapper::Modify(XMLFeature node)
// Changeset id should be updated for every OSM server commit.
node.SetAttribute("changeset", strings::to_string(m_changesetId));
m_api.ModifyElement(node);
m_modified_types[GetTypeForFeature(node)]++;
}
void ChangesetWrapper::Delete(XMLFeature node)
@ -226,4 +258,62 @@ void ChangesetWrapper::Delete(XMLFeature node)
m_api.DeleteElement(node);
}
} // namespace osm
string ChangesetWrapper::TypeCountToString(TTypeCount const & typeCount) const
{
if (typeCount.empty())
return string();
if (typeCount.size() > 3)
{
// TODO(zverik): Instead return top 2 + "and XX other features".
int count = 0;
for (auto const & tc : typeCount)
count += tc.second;
return strings::to_string(count) + " objects";
}
// TODO(zverik): Reverse sort by count
vector<pair<string, uint16_t>> items;
for (auto const & tc : typeCount)
items.push_back(tc);
ostringstream ss;
for (auto i = 0; i < items.size(); ++i)
{
if (i > 0)
{
// Separator: "A and B" for two, "A, B, and C" for three or more.
if (items.size() > 2)
ss << ", ";
else
ss << " ";
if (i == items.size() - 1)
ss << "and ";
}
// Format a count: "a shop" for single shop, "4 shops" for multiple.
if (items[i].second == 1)
ss << "a ";
else
ss << items[i].second << ' ';
ss << items[i].first;
if (items[i].second > 1)
ss << 's';
}
return ss.str();
}
string ChangesetWrapper::GetDescription() const
{
string result;
if (!m_created_types.empty())
result = "Created " + TypeCountToString(m_created_types);
if (!m_modified_types.empty())
{
if (!result.empty())
result += "; ";
result += "Updated " + TypeCountToString(m_modified_types);
}
return result;
}
} // namespace osm

View file

@ -19,6 +19,8 @@ struct ClientToken;
class ChangesetWrapper
{
using TTypeCount = map<string, uint16_t>;
public:
DECLARE_EXCEPTION(ChangesetWrapperException, RootException);
DECLARE_EXCEPTION(NetworkErrorException, ChangesetWrapperException);
@ -32,7 +34,8 @@ public:
DECLARE_EXCEPTION(RelationFeatureAreNotSupportedException, ChangesetWrapperException);
DECLARE_EXCEPTION(EmptyFeatureException, ChangesetWrapperException);
ChangesetWrapper(TKeySecret const & keySecret, ServerApi06::TKeyValueTags const & comments) noexcept;
ChangesetWrapper(TKeySecret const & keySecret,
ServerApi06::TKeyValueTags const & comments) noexcept;
~ChangesetWrapper();
/// Throws many exceptions from above list, plus including XMLNode's parsing ones.
@ -60,6 +63,12 @@ private:
ServerApi06 m_api;
static constexpr uint64_t kInvalidChangesetId = 0;
uint64_t m_changesetId = kInvalidChangesetId;
// No m_deleted_types here, since we do not delete features.
TTypeCount m_modified_types;
TTypeCount m_created_types;
string TypeCountToString(TTypeCount const & typeCount) const;
string GetDescription() const;
};
} // namespace osm
} // namespace osm

View file

@ -12,6 +12,21 @@
#include "3party/pugixml/src/pugixml.hpp"
namespace
{
string KeyValueTagsToXML(osm::ServerApi06::TKeyValueTags const & kvTags)
{
ostringstream stream;
stream << "<osm>\n"
"<changeset>\n";
for (auto const & tag : kvTags)
stream << " <tag k=\"" << tag.first << "\" v=\"" << tag.second << "\"/>\n";
stream << "</changeset>\n"
"</osm>\n";
return stream.str();
}
} // namespace
namespace osm
{
@ -25,15 +40,7 @@ uint64_t ServerApi06::CreateChangeSet(TKeyValueTags const & kvTags) const
if (!m_auth.IsAuthorized())
MYTHROW(NotAuthorized, ("Not authorized."));
ostringstream stream;
stream << "<osm>\n"
"<changeset>\n";
for (auto const & tag : kvTags)
stream << " <tag k=\"" << tag.first << "\" v=\"" << tag.second << "\"/>\n";
stream << "</changeset>\n"
"</osm>\n";
OsmOAuth::Response const response = m_auth.Request("/changeset/create", "PUT", stream.str());
OsmOAuth::Response const response = m_auth.Request("/changeset/create", "PUT", KeyValueTagsToXML(kvTags));
if (response.first != OsmOAuth::HTTP::OK)
MYTHROW(CreateChangeSetHasFailed, ("CreateChangeSet request has failed:", response));
@ -96,6 +103,13 @@ void ServerApi06::DeleteElement(editor::XMLFeature const & element) const
MYTHROW(ErrorDeletingElement, ("Could not delete an element:", response));
}
void ServerApi06::UpdateChangeSet(uint64_t changesetId, TKeyValueTags const & kvTags) const
{
OsmOAuth::Response const response = m_auth.Request("/changeset/" + strings::to_string(changesetId), "PUT", KeyValueTagsToXML(kvTags));
if (response.first != OsmOAuth::HTTP::OK)
MYTHROW(CreateChangeSetHasFailed, ("UpdateChangeSet request has failed:", response));
}
void ServerApi06::CloseChangeSet(uint64_t changesetId) const
{
OsmOAuth::Response const response = m_auth.Request("/changeset/" + strings::to_string(changesetId) + "/close", "PUT");

View file

@ -70,6 +70,7 @@ public:
/// @param element should already have all attributes set, including "id", "version", "changeset".
/// @returns true if element was successfully deleted (or was already deleted).
void DeleteElement(editor::XMLFeature const & element) const;
void UpdateChangeSet(uint64_t changesetId, TKeyValueTags const & kvTags) const;
void CloseChangeSet(uint64_t changesetId) const;
/// @returns id of a created note.
uint64_t CreateNote(ms::LatLon const & ll, string const & message) const;