New version of UGCUpdate

This commit is contained in:
VladiMihaylenko 2017-11-03 17:50:08 +03:00 committed by r.kuznetsov
parent 27a015e6aa
commit 6c389a2e4c
9 changed files with 321 additions and 83 deletions

View file

@ -12,6 +12,7 @@
#include <cmath>
#include <cstdint>
#include <vector>
namespace ugc
{
@ -21,7 +22,8 @@ DECLARE_EXCEPTION(BadBlob, SerDesException);
enum class Version : uint8_t
{
V0,
Latest = V0
V1,
Latest = V1
};
template <typename Sink>
@ -50,6 +52,11 @@ public:
WriteToSink(m_sink, index);
}
void VisitLangs(std::vector<uint8_t> const & indexes, char const * /* name */ = nullptr)
{
(*this)(indexes);
}
template <typename T>
void VisitVarUint(T const & t, char const * /* name */ = nullptr)
{
@ -127,6 +134,11 @@ public:
ReadPrimitiveFromSource<uint8_t>(m_source, index);
}
void VisitLangs(std::vector<uint8_t> & indexes, char const * /* name */ = nullptr)
{
(*this)(indexes);
}
template <typename T>
void VisitVarUint(T & t, char const * /* name */ = nullptr)
{
@ -177,20 +189,46 @@ private:
Source & m_source;
};
// Version must be used only for testing.
// We should use only the latest version in production.
template <typename Sink, typename UGC>
void Serialize(Sink & sink, UGC const & ugc)
void Serialize(Sink & sink, UGC const & ugc, Version const version = Version::Latest)
{
WriteToSink(sink, static_cast<uint8_t>(Version::Latest));
WriteToSink(sink, static_cast<uint8_t>(version));
Serializer<Sink> ser(sink);
ser(ugc);
}
template <typename Source, typename UGC>
template <typename Source>
void Deserialize(Source & source, UGC & ugc)
{
uint8_t version = 0;
ReadPrimitiveFromSource(source, version);
if (version == static_cast<uint8_t>(Version::V0) || version == static_cast<uint8_t>(Version::V1))
{
DeserializerV0<Source> des(source);
des(ugc);
return;
}
MYTHROW(BadBlob, ("Unknown data version:", static_cast<int>(version)));
}
template <typename Source>
void Deserialize(Source & source, UGCUpdate & ugc)
{
uint8_t version = 0;
ReadPrimitiveFromSource(source, version);
if (version == static_cast<uint8_t>(Version::V0))
{
DeserializerV0<Source> des(source);
v0::UGCUpdate old;
des(old);
ugc.BuildFrom(old);
return;
}
if (version == static_cast<uint8_t>(Version::V1))
{
DeserializerV0<Source> des(source);
des(ugc);

View file

@ -8,6 +8,7 @@
#include <cstdint>
#include <cstdlib>
#include <vector>
namespace ugc
{
@ -85,6 +86,15 @@ public:
ToJSONObject(*m_json, name, StringUtf8Multilang::GetLangByCode(index));
}
void VisitLangs(std::vector<uint8_t> const & indexes, char const * name = nullptr)
{
std::vector<std::string> langs;
for (auto const index : indexes)
langs.emplace_back(StringUtf8Multilang::GetLangByCode(index));
ToJSONObject(*m_json, name, langs);
}
void VisitPoint(m2::PointD const & pt, char const * x = nullptr, char const * y = nullptr)
{
(*this)(pt.x, x);
@ -214,6 +224,14 @@ public:
index = StringUtf8Multilang::GetLangIndex(lang);
}
void VisitLangs(std::vector<uint8_t> & indexes, char const * name = nullptr)
{
std::vector<std::string> langs;
FromJSONObject(m_json, name, langs);
for (auto const & lang : langs)
indexes.emplace_back(StringUtf8Multilang::GetLangIndex(lang));
}
void VisitPoint(m2::PointD & pt, char const * x = nullptr, char const * y = nullptr)
{
FromJSONObject(m_json, x, pt.x);

View file

@ -72,6 +72,61 @@ string SerializeUGCIndex(vector<Storage::UGCIndex> const & indexes)
unique_ptr<char, JSONFreeDeleter> buffer(json_dumps(array.get(), JSON_COMPACT | JSON_ENSURE_ASCII));
return string(buffer.get());
}
template <typename UGCUpdate>
Storage::SettingResult SetGenericUGCUpdate(
vector<Storage::UGCIndex> & indexes, size_t & numberOfDeleted, FeatureID const & id,
UGCUpdate const & ugc,
FeatureType const & featureType,
Version const version = Version::Latest)
{
if (!ugc.IsValid())
return Storage::SettingResult::InvalidUGC;
auto const mercator = feature::GetCenter(featureType);
feature::TypesHolder th(featureType);
th.SortBySpec();
auto const optMatchingType = ftraits::UGC::GetType(th);
CHECK(optMatchingType, ());
auto const type = th.GetBestType();
for (auto & index : indexes)
{
if (type == index.m_type && mercator == index.m_mercator && !index.m_deleted)
{
index.m_deleted = true;
++numberOfDeleted;
break;
}
}
Storage::UGCIndex index;
uint64_t offset;
if (!GetUGCFileSize(offset))
offset = 0;
index.m_mercator = mercator;
index.m_type = type;
index.m_matchingType = *optMatchingType;
index.m_mwmName = id.GetMwmName();
index.m_dataVersion = id.GetMwmVersion();
index.m_featureId = id.m_index;
index.m_offset = offset;
auto const ugcFilePath = GetUGCFilePath();
try
{
FileWriter w(ugcFilePath, FileWriter::Op::OP_APPEND);
Serialize(w, ugc, version);
indexes.emplace_back(move(index));
}
catch (FileWriter::Exception const & exception)
{
LOG(LERROR, ("Exception while writing file:", ugcFilePath, "reason:", exception.what()));
return Storage::SettingResult::WritingError;
}
return Storage::SettingResult::Success;
}
} // namespace
UGCUpdate Storage::GetUGCUpdate(FeatureID const & id) const
@ -105,7 +160,7 @@ UGCUpdate Storage::GetUGCUpdate(FeatureID const & id) const
}
catch (FileReader::Exception const & exception)
{
LOG(LERROR, ("Exception while reading file:", ugcFilePath, "reason:", exception.Msg()));
LOG(LERROR, ("Exception while reading file:", ugcFilePath, "reason:", exception.what()));
return {};
}
@ -118,54 +173,9 @@ UGCUpdate Storage::GetUGCUpdate(FeatureID const & id) const
Storage::SettingResult Storage::SetUGCUpdate(FeatureID const & id, UGCUpdate const & ugc)
{
if (!ugc.IsValid())
return Storage::SettingResult::InvalidUGC;
auto const feature = GetFeature(id);
auto const mercator = feature::GetCenter(*feature);
feature::TypesHolder th(*feature);
th.SortBySpec();
auto const optMatchingType = ftraits::UGC::GetType(th);
CHECK(optMatchingType, ());
auto const type = th.GetBestType();
for (auto & index : m_UGCIndexes)
{
if (type == index.m_type && mercator == index.m_mercator && !index.m_deleted)
{
index.m_deleted = true;
m_numberOfDeleted++;
break;
}
}
// TODO: Call Defragmentation().
UGCIndex index;
uint64_t offset;
if (!GetUGCFileSize(offset))
offset = 0;
index.m_mercator = mercator;
index.m_type = type;
index.m_matchingType = *optMatchingType;
index.m_mwmName = id.GetMwmName();
index.m_dataVersion = id.GetMwmVersion();
index.m_featureId = id.m_index;
index.m_offset = offset;
auto const ugcFilePath = GetUGCFilePath();
try
{
FileWriter w(ugcFilePath, FileWriter::Op::OP_APPEND);
Serialize(w, ugc);
m_UGCIndexes.emplace_back(move(index));
}
catch (FileWriter::Exception const & exception)
{
LOG(LERROR, ("Exception while writing file:", ugcFilePath, "reason:", exception.Msg()));
return Storage::SettingResult::WritingError;
}
return Storage::SettingResult::Success;
return SetGenericUGCUpdate(m_UGCIndexes, m_numberOfDeleted, id, ugc,
*feature);
}
void Storage::Load()
@ -179,7 +189,7 @@ void Storage::Load()
}
catch (FileReader::Exception const & exception)
{
LOG(LWARNING, ("Exception while reading file:", indexFilePath, "reason:", exception.Msg()));
LOG(LWARNING, ("Exception while reading file:", indexFilePath, "reason:", exception.what()));
return;
}
@ -205,7 +215,7 @@ void Storage::SaveIndex() const
}
catch (FileWriter::Exception const & exception)
{
LOG(LERROR, ("Exception while writing file:", indexFilePath, "reason:", exception.Msg()));
LOG(LERROR, ("Exception while writing file:", indexFilePath, "reason:", exception.what()));
}
}
@ -241,12 +251,12 @@ void Storage::Defragmentation()
}
catch (FileReader::Exception const & exception)
{
LOG(LERROR, ("Exception while reading file:", ugcFilePath, "reason:", exception.Msg()));
LOG(LERROR, ("Exception while reading file:", ugcFilePath, "reason:", exception.what()));
return;
}
catch (FileWriter::Exception const & exception)
{
LOG(LERROR, ("Exception while writing file:", tmpUGCFilePath, "reason:", exception.Msg()));
LOG(LERROR, ("Exception while writing file:", tmpUGCFilePath, "reason:", exception.what()));
return;
}
@ -285,7 +295,7 @@ string Storage::GetUGCToSend() const
}
catch (FileReader::Exception const & exception)
{
LOG(LERROR, ("Exception while reading file:", ugcFilePath, "reason:", exception.Msg()));
LOG(LERROR, ("Exception while reading file:", ugcFilePath, "reason:", exception.what()));
return string();
}
@ -371,4 +381,14 @@ unique_ptr<FeatureType> Storage::GetFeature(FeatureID const & id) const
CHECK(feature, ());
return feature;
}
// Testing
Storage::SettingResult Storage::SetUGCUpdateForTesting(FeatureID const & id,
v0::UGCUpdate const & ugc)
{
auto const feature = GetFeature(id);
return SetGenericUGCUpdate(m_UGCIndexes, m_numberOfDeleted, id, ugc,
*feature, Version::V0);
}
} // namespace ugc

View file

@ -60,6 +60,7 @@ public:
/// Testing
std::vector<UGCIndex> const & GetIndexesForTesting() const { return m_UGCIndexes; }
size_t GetNumberOfDeletedForTesting() const { return m_numberOfDeleted; }
SettingResult SetUGCUpdateForTesting(FeatureID const & id, v0::UGCUpdate const & ugc);
private:
uint64_t UGCSizeAtIndex(size_t const indexPosition) const;

View file

@ -140,6 +140,42 @@ inline std::string DebugPrint(Text const & text)
return os.str();
}
struct KeyboardText
{
KeyboardText() = default;
KeyboardText(std::string const & text, uint8_t const deviceLang,
std::vector<uint8_t> const & keyboardLangs)
: m_text(text), m_deviceLang(deviceLang), m_keyboardLangs(keyboardLangs)
{
}
DECLARE_VISITOR(visitor(m_text, "text"), visitor.VisitLang(m_deviceLang, "lang"),
visitor.VisitLangs(m_keyboardLangs, "kbd_langs"))
bool operator==(KeyboardText const & rhs) const
{
return m_text == rhs.m_text && m_deviceLang == rhs.m_deviceLang &&
m_keyboardLangs == rhs.m_keyboardLangs;
}
std::string m_text;
uint8_t m_deviceLang = StringUtf8Multilang::kDefaultCode;
std::vector<uint8_t> m_keyboardLangs;
};
inline std::string DebugPrint(KeyboardText const & text)
{
std::ostringstream os;
os << "Keyboard text [ " << StringUtf8Multilang::GetLangByCode(text.m_deviceLang) << ": "
<< text.m_text << " ]";
os << "Keyboard locales [ ";
for (auto const index : text.m_keyboardLangs)
os << StringUtf8Multilang::GetLangByCode(index) << " ";
os << " ]";
return os.str();
}
struct Review
{
using ReviewId = uint64_t;
@ -199,6 +235,84 @@ struct Attribute
TranslationKey m_value{};
};
struct ReviewFeedback
{
ReviewFeedback() = default;
ReviewFeedback(Sentiment const sentiment, Time const & time)
: m_sentiment(sentiment), m_time(time)
{
}
Sentiment m_sentiment{};
Time m_time{};
};
struct ReviewAbuse
{
ReviewAbuse() = default;
ReviewAbuse(std::string const & reason, Time const & time) : m_reason(reason), m_time(time) {}
std::string m_reason{};
Time m_time{};
};
namespace v0
{
struct UGCUpdate
{
UGCUpdate() = default;
UGCUpdate(Ratings const & records, Text const & text, Time const & time)
: m_ratings(records), m_text(text), m_time(time)
{
}
DECLARE_VISITOR(visitor(m_ratings, "ratings"), visitor(m_text, "text"), visitor(m_time, "date"))
bool operator==(UGCUpdate const & rhs) const
{
return m_ratings == rhs.m_ratings && m_text == rhs.m_text && m_time == rhs.m_time;
}
bool IsEmpty() const
{
return (m_ratings.empty() && m_text.m_text.empty()) || m_time == Time();
}
bool IsValid() const
{
bool const timeIsValid = m_time != Time();
if (!m_text.m_text.empty())
return timeIsValid;
bool ratingIsValid = false;
for (auto const & r : m_ratings)
{
if (static_cast<int>(r.m_value) > 0)
{
ratingIsValid = true;
break;
}
}
return ratingIsValid && timeIsValid;
}
Ratings m_ratings;
Text m_text;
Time m_time{};
};
inline std::string DebugPrint(UGCUpdate const & ugcUpdate)
{
std::ostringstream os;
os << "UGCUpdate [ ";
os << "records:" << ::DebugPrint(ugcUpdate.m_ratings) << ", ";
os << "text:" << DebugPrint(ugcUpdate.m_text) << ", ";
os << "days since epoch:" << ToDaysSinceEpoch(ugcUpdate.m_time) << " ]";
return os.str();
}
} // namespace v0
struct UGC
{
UGC() = default;
@ -242,7 +356,7 @@ inline std::string DebugPrint(UGC const & ugc)
struct UGCUpdate
{
UGCUpdate() = default;
UGCUpdate(Ratings const & records, Text const & text, Time const & time)
UGCUpdate(Ratings const & records, KeyboardText const & text, Time const & time)
: m_ratings(records), m_text(text), m_time(time)
{
}
@ -254,9 +368,17 @@ struct UGCUpdate
return m_ratings == rhs.m_ratings && m_text == rhs.m_text && m_time == rhs.m_time;
}
void BuildFrom(v0::UGCUpdate const & update)
{
m_ratings = update.m_ratings;
m_text.m_text = update.m_text.m_text;
m_text.m_deviceLang = update.m_text.m_lang;
m_time = update.m_time;
}
bool IsEmpty() const
{
return !((!m_ratings.empty() || !m_text.m_text.empty()) && m_time != Time());
return (m_ratings.empty() && m_text.m_text.empty()) || m_time == Time();
}
bool IsValid() const
@ -279,7 +401,7 @@ struct UGCUpdate
}
Ratings m_ratings;
Text m_text;
KeyboardText m_text;
Time m_time{};
};
@ -292,26 +414,5 @@ inline std::string DebugPrint(UGCUpdate const & ugcUpdate)
os << "days since epoch:" << ToDaysSinceEpoch(ugcUpdate.m_time) << " ]";
return os.str();
}
struct ReviewFeedback
{
ReviewFeedback() = default;
ReviewFeedback(Sentiment const sentiment, Time const & time)
: m_sentiment(sentiment), m_time(time)
{
}
Sentiment m_sentiment{};
Time m_time{};
};
struct ReviewAbuse
{
ReviewAbuse() = default;
ReviewAbuse(std::string const & reason, Time const & time) : m_reason(reason), m_time(time) {}
std::string m_reason{};
Time m_time{};
};
} // namespace ugc

View file

@ -149,4 +149,26 @@ UNIT_TEST(SerDes_UGCUpdate)
TEST_EQUAL(expectedUGC, actualUGC, ());
}
UNIT_TEST(SerDes_UGCUpdateDifferentVersions)
{
auto const v0 = MakeTestUGCUpdateV0(Time(chrono::hours(24 * 10)));
TEST_EQUAL(v0, v0, ());
Buffer buffer;
{
auto sink = MakeSink(buffer);
Serialize(sink, v0, Version::V0);
}
UGCUpdate actualUGC{};
{
auto source = MakeSource(buffer);
Deserialize(source, actualUGC);
}
UGCUpdate expectedUgc;
expectedUgc.BuildFrom(v0);
TEST_EQUAL(expectedUgc, actualUGC, ());
}
} // namespace

View file

@ -338,7 +338,7 @@ UNIT_CLASS_TEST(StorageTest, InvalidUGC)
TEST_EQUAL(storage.SetUGCUpdate(cafeId, first), Storage::SettingResult::InvalidUGC, ());
TEST(storage.GetIndexesForTesting().empty(), ());
TEST(storage.GetUGCToSend().empty(), ());
first.m_text = Text("a", 1);
first.m_text = KeyboardText("a", 1, {2});
TEST_EQUAL(storage.SetUGCUpdate(cafeId, first), Storage::SettingResult::Success, ());
UGCUpdate second;
second.m_time = chrono::system_clock::now();
@ -347,3 +347,26 @@ UNIT_CLASS_TEST(StorageTest, InvalidUGC)
second.m_ratings.emplace_back("b", 1);
TEST_EQUAL(storage.SetUGCUpdate(cafeId, second), Storage::SettingResult::Success, ());
}
UNIT_CLASS_TEST(StorageTest, DifferentUGCVersions)
{
auto & builder = MwmBuilder::Builder();
m2::PointD const firstPoint(1.0, 1.0);
m2::PointD const secondPoint(2.0, 2.0);
builder.Build({TestCafe(firstPoint), TestCafe(secondPoint)});
Storage storage(builder.GetIndex());
storage.Load();
auto const firstId = builder.FeatureIdForCafeAtPoint(firstPoint);
auto const oldUGC = MakeTestUGCUpdateV0(Time(chrono::hours(24 * 10)));
TEST_EQUAL(storage.SetUGCUpdateForTesting(firstId, oldUGC), Storage::SettingResult::Success, ());
auto const secondId = builder.FeatureIdForCafeAtPoint(secondPoint);
auto const newUGC = MakeTestUGCUpdate(Time(chrono::hours(24 * 5)));
TEST_EQUAL(storage.SetUGCUpdate(secondId, newUGC), Storage::SettingResult::Success, ());
TEST_EQUAL(newUGC, storage.GetUGCUpdate(secondId), ());
UGCUpdate fromOld;
fromOld.BuildFrom(oldUGC);
TEST_EQUAL(fromOld, storage.GetUGCUpdate(firstId), ());
}

View file

@ -43,7 +43,7 @@ UGC MakeTestUGC2(Time now)
return UGC(records, reviews, 5.0 /* rating */, 1 /* votes */);
}
UGCUpdate MakeTestUGCUpdate(Time now)
v0::UGCUpdate MakeTestUGCUpdateV0(Time now)
{
Ratings records;
records.emplace_back("food" /* key */, 4.0 /* value */);
@ -54,6 +54,20 @@ UGCUpdate MakeTestUGCUpdate(Time now)
Time time{FromDaysAgo(now, 1)};
return v0::UGCUpdate(records, text, time);
}
UGCUpdate MakeTestUGCUpdate(Time now)
{
Ratings records;
records.emplace_back("food" /* key */, 4.0 /* value */);
records.emplace_back("service" /* key */, 5.0 /* value */);
records.emplace_back("music" /* key */, 5.0 /* value */);
KeyboardText text{"It's aways nice to visit this place", StringUtf8Multilang::kEnglishCode, {1}};
Time time{FromDaysAgo(now, 1)};
return UGCUpdate(records, text, time);
}
}

View file

@ -7,5 +7,6 @@ namespace ugc
UGC MakeTestUGC1(Time now = Clock::now());
UGC MakeTestUGC2(Time now = Clock::now());
v0::UGCUpdate MakeTestUGCUpdateV0(Time now = Clock::now());
UGCUpdate MakeTestUGCUpdate(Time now = Clock::now());
} // namespace ugc