[ugc] HasUGCForPlace method is added

This commit is contained in:
Arsentiy Milchakov 2018-11-13 15:58:52 +03:00 committed by Vlad Mihaylenko
parent 4fa50bb98d
commit 703971ad11
6 changed files with 88 additions and 13 deletions

View file

@ -1,7 +1,5 @@
#pragma once
#include "storage/index.hpp"
#include "geometry/mercator.hpp"
#include "geometry/point2d.hpp"
#include "geometry/tree4d.hpp"

View file

@ -1,5 +1,7 @@
#include "ugc/api.hpp"
#include "indexer/feature_data.hpp"
#include "base/assert.hpp"
#include <utility>
@ -38,6 +40,16 @@ void Api::GetUGCToSend(UGCJsonToSendCallback const & callback)
m_thread.Push([callback, this] { GetUGCToSendImpl(callback); });
}
void Api::HasUGCForPlace(feature::TypesHolder const & types, m2::PointD const & point,
HasUGCForPlaceCallback const & callback)
{
m_thread.Push([this, editableTypes = types, point, callback]() mutable
{
editableTypes.SortBySpec();
HasUGCForPlaceImpl(editableTypes.GetBestType(), point, callback);
});
}
void Api::SendingCompleted()
{
m_thread.Push([this] { SendingCompletedImpl(); });
@ -80,6 +92,12 @@ void Api::GetUGCToSendImpl(UGCJsonToSendCallback const & callback)
callback(move(json), m_storage.GetNumberOfUnsynchronized());
}
void Api::HasUGCForPlaceImpl(uint32_t bestType, m2::PointD const & point,
HasUGCForPlaceCallback const & callback) const
{
callback(m_storage.HasUGCForPlace(bestType, point));
}
void Api::SendingCompletedImpl()
{
m_storage.MarkAllAsSynchronized();

View file

@ -6,6 +6,8 @@
#include "platform/safe_callback.hpp"
#include "geometry/point2d.hpp"
#include "base/worker_thread.hpp"
#include <functional>
@ -13,6 +15,11 @@
class DataSource;
struct FeatureID;
namespace feature
{
class TypesHolder;
}
namespace ugc
{
class Api
@ -23,6 +30,7 @@ public:
using UGCJsonToSendCallback = std::function<void(std::string && jsonStr, size_t numberOfUnsynchronized)>;
using OnResultCallback = platform::SafeCallback<void(Storage::SettingResult const result)>;
using NumberOfUnsynchronizedCallback = std::function<void(size_t number)>;
using HasUGCForPlaceCallback = std::function<void(bool result)>;
Api(DataSource const & dataSource, NumberOfUnsynchronizedCallback const & callback);
@ -30,6 +38,8 @@ public:
void SetUGCUpdate(FeatureID const & id, UGCUpdate const & ugc,
OnResultCallback const & callback = nullptr);
void GetUGCToSend(UGCJsonToSendCallback const & callback);
void HasUGCForPlace(feature::TypesHolder const & types, m2::PointD const & point,
HasUGCForPlaceCallback const & callback);
void SendingCompleted();
void SaveUGCOnDisk();
@ -39,6 +49,8 @@ private:
void GetUGCImpl(FeatureID const & id, UGCCallbackUnsafe const & callback);
Storage::SettingResult SetUGCUpdateImpl(FeatureID const & id, UGCUpdate const & ugc);
void GetUGCToSendImpl(UGCJsonToSendCallback const & callback);
void HasUGCForPlaceImpl(uint32_t bestType, m2::PointD const & point,
HasUGCForPlaceCallback const & callback) const;
void SendingCompletedImpl();
void SaveUGCOnDiskImpl();

View file

@ -7,6 +7,7 @@
#include "indexer/classificator.hpp"
#include "indexer/feature_algo.hpp"
#include "indexer/feature_data.hpp"
#include "indexer/feature_decl.hpp"
#include "indexer/ftraits.hpp"
@ -155,17 +156,7 @@ UGCUpdate Storage::GetUGCUpdate(FeatureID const & id) const
if (m_indexes.empty())
return {};
auto const feature = GetFeature(id);
auto const mercator = feature::GetCenter(*feature);
feature::TypesHolder th(*feature);
th.SortBySpec();
auto const & c = classif();
auto const type = c.GetIndexForType(th.GetBestType());
auto const index = find_if(
m_indexes.begin(), m_indexes.end(), [type, &mercator](UpdateIndex const & index) -> bool {
return type == index.m_type && mercator == index.m_mercator && !index.m_deleted;
});
auto const index = FindIndex(id);
if (index == m_indexes.end())
return {};
@ -291,6 +282,30 @@ void Storage::Migrate(string const & indexFilePath)
}
}
UpdateIndexes::const_iterator Storage::FindIndex(FeatureID const & id) const
{
auto const feature = GetFeature(id);
auto const mercator = feature::GetCenter(*feature);
feature::TypesHolder th(*feature);
th.SortBySpec();
return FindIndex(th.GetBestType(), mercator);
}
UpdateIndexes::const_iterator Storage::FindIndex(uint32_t bestType,
m2::PointD const & point) const
{
auto const & c = classif();
auto const typeIndex = c.GetIndexForType(bestType);
return find_if(
m_indexes.begin(), m_indexes.end(), [typeIndex, &point](UpdateIndex const & index) -> bool {
// We are use 1e-5 eps because of points in mwm have this accuracy.
return typeIndex == index.m_type && point.EqualDxDy(index.m_mercator, 1e-5 /* eps */) &&
!index.m_deleted;
});
}
bool Storage::SaveIndex(std::string const & pathToTargetFile /* = "" */) const
{
if (m_indexes.empty())
@ -442,6 +457,11 @@ size_t Storage::GetNumberOfUnsynchronized() const
return numberOfUnsynchronized;
}
bool Storage::HasUGCForPlace(uint32_t bestType, m2::PointD const & point) const
{
return FindIndex(bestType, point) != m_indexes.end();
}
void Storage::MarkAllAsSynchronized()
{
if (m_indexes.empty())

View file

@ -2,6 +2,8 @@
#include "ugc/types.hpp"
#include "geometry/point2d.hpp"
#include "base/thread_checker.hpp"
#include <memory>
@ -11,6 +13,11 @@ class DataSource;
class FeatureType;
struct FeatureID;
namespace feature
{
class TypesHolder;
}
namespace ugc
{
class Storage
@ -34,6 +41,7 @@ public:
void Defragmentation();
void Load();
size_t GetNumberOfUnsynchronized() const;
bool HasUGCForPlace(uint32_t bestType, m2::PointD const & point) const;
/// Testing
UpdateIndexes & GetIndexesForTesting() { return m_indexes; }
@ -46,6 +54,8 @@ private:
uint64_t UGCSizeAtIndex(size_t const indexPosition) const;
std::unique_ptr<FeatureType> GetFeature(FeatureID const & id) const;
void Migrate(std::string const & indexFilePath);
UpdateIndexes::const_iterator FindIndex(FeatureID const & id) const;
UpdateIndexes::const_iterator FindIndex(uint32_t bestType, m2::PointD const & point) const;
DataSource const & m_dataSource;
UpdateIndexes m_indexes;

View file

@ -604,3 +604,20 @@ UNIT_TEST(UGC_TooOldDataVersionsForMigration)
TEST(DeleteUGCFile(IndexVersion::V0), ());
}
UNIT_CLASS_TEST(StorageTest, UGC_HasUGCForPlace)
{
auto & builder = MwmBuilder::Builder();
m2::PointD const point(1.0, 1.0);
builder.Build({TestCafe(point)});
auto const id = builder.FeatureIdForCafeAtPoint(point);
auto const original = MakeTestUGCUpdate(Time(chrono::hours(24 * 300)));
Storage storage(builder.GetDataSource());
storage.Load();
TEST_EQUAL(storage.SetUGCUpdate(id, original), Storage::SettingResult::Success, ());
auto const actual = storage.GetUGCUpdate(id);
TEST_EQUAL(original, actual, ());
auto const & c = classif();
auto const cafeType = c.GetTypeByReadableObjectName("amenity-cafe");
TEST(storage.HasUGCForPlace(cafeType, point), ());
}