[search] Usage of CitiesBoundaries in LocalityFinder.

This commit is contained in:
Yuri Gorshenin 2017-10-17 15:16:26 +03:00 committed by Tatiana Yan
parent c002ee258b
commit 18caeab025
21 changed files with 217 additions and 58 deletions

View file

@ -232,6 +232,12 @@ public:
ForEachInIntervals(implFunctor, covering::FullCover, m2::RectD::GetInfiniteRect(), scale);
}
template <typename F>
void ReadFeature(F && f, FeatureID const & feature) const
{
return ReadFeatures(forward<F>(f), {feature});
}
// "features" must be sorted using FeatureID::operator< as predicate.
template <typename F>
void ReadFeatures(F && f, std::vector<FeatureID> const & features) const

View file

@ -30,14 +30,15 @@ bool CitiesBoundariesTable::Boundaries::HasPoint(m2::PointD const & p) const
}
// CitiesBoundariesTable ---------------------------------------------------------------------------
bool CitiesBoundariesTable::Load(Index const & index)
bool CitiesBoundariesTable::Load()
{
auto handle = FindWorld(index);
auto handle = FindWorld(m_index);
if (!handle.IsAlive())
{
LOG(LERROR, ("Can't find world map file"));
return false;
}
// Skip if table was already loaded from this file.
if (handle.GetId() == m_mwmId)
return true;
MwmContext context(move(handle));
auto const localities = CategoriesCache(LocalitiesSource{}, my::Cancellable{}).Get(context);
@ -73,6 +74,7 @@ bool CitiesBoundariesTable::Load(Index const & index)
return false;
}
m_mwmId = context.GetId();
m_table.clear();
m_eps = precision;
size_t boundary = 0;
@ -85,6 +87,13 @@ bool CitiesBoundariesTable::Load(Index const & index)
return true;
}
bool CitiesBoundariesTable::Get(FeatureID const & fid, Boundaries & bs) const
{
if (fid.m_mwmId != m_mwmId)
return false;
return Get(fid.m_index, bs);
}
bool CitiesBoundariesTable::Get(uint32_t fid, Boundaries & bs) const
{
auto const it = m_table.find(fid);

View file

@ -1,6 +1,7 @@
#pragma once
#include "indexer/city_boundary.hpp"
#include "indexer/feature_decl.hpp"
#include "geometry/point2d.hpp"
@ -11,6 +12,11 @@
class Index;
namespace feature
{
struct FeatureID;
}
namespace search
{
class CitiesBoundariesTable
@ -40,12 +46,19 @@ public:
double m_eps = 0.0;
};
bool Load(Index const & index);
CitiesBoundariesTable(Index const & index) : m_index(index) {}
bool Load();
bool Has(FeatureID const & fid) const { return fid.m_mwmId == m_mwmId && Has(fid.m_index); }
bool Has(uint32_t fid) const { return m_table.find(fid) != m_table.end(); }
bool Get(FeatureID const & fid, Boundaries & bs) const;
bool Get(uint32_t fid, Boundaries & bs) const;
private:
Index const & m_index;
MwmSet::MwmId m_mwmId;
std::unordered_map<uint32_t, std::vector<indexer::CityBoundary>> m_table;
double m_eps = 0.0;
};

View file

@ -14,8 +14,13 @@ namespace search
class CityFinder
{
public:
// TODO (@milchakov): consider to reuse locality finder from search
// engine. Otherwise, CityFinder won't benefit from approximated
// cities boundaries.
explicit CityFinder(Index const & index)
: m_unusedCache(m_cancellable), m_finder(index, m_unusedCache)
: m_unusedBoundaries(index)
, m_unusedCache(m_cancellable)
, m_finder(index, m_unusedBoundaries, m_unusedCache)
{
}
@ -29,6 +34,7 @@ public:
private:
my::Cancellable m_cancellable;
search::CitiesBoundariesTable m_unusedBoundaries;
search::VillagesCache m_unusedCache;
search::LocalityFinder m_finder;
};

View file

@ -110,6 +110,8 @@ Engine::Engine(Index & index, CategoriesHolder const & categories,
m_threads.reserve(params.m_numThreads);
for (size_t i = 0; i < params.m_numThreads; ++i)
m_threads.emplace_back(&Engine::MainLoop, this, ref(m_contexts[i]));
LoadCitiesBoundaries();
}
Engine::~Engine()
@ -150,6 +152,12 @@ void Engine::ClearCaches()
});
}
void Engine::LoadCitiesBoundaries()
{
PostMessage(Message::TYPE_BROADCAST,
[this](Processor & processor) { processor.LoadCitiesBoundaries(); });
}
void Engine::MainLoop(Context & context)
{
while (true)

View file

@ -106,6 +106,10 @@ public:
// Posts request to clear caches to the queue.
void ClearCaches();
// Posts request to reload cities boundaries tables. Must be used
// for testing only.
void LoadCitiesBoundariesForTesting() { return LoadCitiesBoundaries(); }
private:
struct Message
{
@ -140,6 +144,8 @@ private:
unique_ptr<Processor> m_processor;
};
void LoadCitiesBoundaries();
// *ALL* following methods are executed on the m_threads threads.
// This method executes tasks from a common pool (|tasks|) in a FIFO

View file

@ -57,9 +57,11 @@ private:
class LocalitiesLoader
{
public:
LocalitiesLoader(MwmContext const & ctx, Filter const & filter, LocalityFinder::Holder & holder,
LocalitiesLoader(MwmContext const & ctx, CitiesBoundariesTable const & boundaries,
Filter const & filter, LocalityFinder::Holder & holder,
map<MwmSet::MwmId, unordered_set<uint32_t>> & loadedIds)
: m_ctx(ctx)
, m_boundaries(boundaries)
, m_filter(filter)
, m_holder(holder)
, m_loadedIds(loadedIds[m_ctx.GetId()])
@ -99,12 +101,16 @@ public:
auto const names = ft.GetNames();
auto const center = ft.GetCenter();
m_holder.Add(LocalityItem(names, center, population));
CitiesBoundariesTable::Boundaries boundaries;
m_boundaries.Get(FeatureID(m_ctx.GetId(), id), boundaries);
m_holder.Add(LocalityItem(names, center, boundaries, population));
m_loadedIds.insert(id);
}
private:
MwmContext const & m_ctx;
CitiesBoundariesTable const & m_boundaries;
Filter const & m_filter;
LocalityFinder::Holder & m_holder;
@ -114,8 +120,8 @@ private:
// LocalityItem ------------------------------------------------------------------------------------
LocalityItem::LocalityItem(StringUtf8Multilang const & names, m2::PointD const & center,
uint64_t population)
: m_names(names), m_center(center), m_population(population)
Boundaries const & boundaries, uint64_t population)
: m_names(names), m_center(center), m_boundaries(boundaries), m_population(population)
{
}
@ -130,19 +136,27 @@ string DebugPrint(LocalityItem const & item)
// LocalitySelector --------------------------------------------------------------------------------
LocalitySelector::LocalitySelector(m2::PointD const & p) : m_p(p) {}
void LocalitySelector::operator()(LocalityItem const & item)
{
auto const inside = item.m_boundaries.HasPoint(m_p);
// TODO (@y, @m): replace this naive score by p-values on
// multivariate Gaussian.
double const distance = MercatorBounds::DistanceOnEarth(item.m_center, m_p);
double const score =
ftypes::GetPopulationByRadius(distance) / static_cast<double>(item.m_population);
if (score < m_bestScore)
if (!inside && m_inside)
return;
ASSERT(inside || !m_inside, ());
if ((inside && !m_inside) || (score < m_score))
{
m_bestScore = score;
m_bestLocality = &item;
m_inside = inside;
m_score = score;
m_locality = &item;
}
}
@ -189,8 +203,10 @@ void LocalityFinder::Holder::Clear()
}
// LocalityFinder ----------------------------------------------------------------------------------
LocalityFinder::LocalityFinder(Index const & index, VillagesCache & villagesCache)
LocalityFinder::LocalityFinder(Index const & index, CitiesBoundariesTable const & boundariesTable,
VillagesCache & villagesCache)
: m_index(index)
, m_boundariesTable(boundariesTable)
, m_villagesCache(villagesCache)
, m_cities(kMaxCityRadiusMeters)
, m_villages(kMaxVillageRadiusMeters)
@ -228,7 +244,8 @@ void LocalityFinder::LoadVicinity(m2::PointD const & p, bool loadCities, bool lo
m_ranks = make_unique<DummyRankTable>();
MwmContext ctx(move(handle));
ctx.ForEachIndex(crect, LocalitiesLoader(ctx, CityFilter(*m_ranks), m_cities, m_loadedIds));
ctx.ForEachIndex(crect, LocalitiesLoader(ctx, m_boundariesTable, CityFilter(*m_ranks),
m_cities, m_loadedIds));
}
m_cities.SetCovered(p);
@ -243,8 +260,9 @@ void LocalityFinder::LoadVicinity(m2::PointD const & p, bool loadCities, bool lo
return;
MwmContext ctx(move(handle));
ctx.ForEachIndex(vrect, LocalitiesLoader(ctx, VillageFilter(ctx, m_villagesCache), m_villages,
m_loadedIds));
ctx.ForEachIndex(vrect,
LocalitiesLoader(ctx, m_boundariesTable, VillageFilter(ctx, m_villagesCache),
m_villages, m_loadedIds));
});
m_villages.SetCovered(p);

View file

@ -1,5 +1,7 @@
#pragma once
#include "search/cities_boundaries_table.hpp"
#include "indexer/mwm_set.hpp"
#include "indexer/rank_table.hpp"
@ -26,7 +28,10 @@ class VillagesCache;
struct LocalityItem
{
LocalityItem(StringUtf8Multilang const & names, m2::PointD const & center, uint64_t population);
using Boundaries = CitiesBoundariesTable::Boundaries;
LocalityItem(StringUtf8Multilang const & names, m2::PointD const & center,
Boundaries const & boundaries, uint64_t population);
bool GetName(int8_t lang, string & name) const { return m_names.GetString(lang, name); }
@ -37,6 +42,7 @@ struct LocalityItem
StringUtf8Multilang m_names;
m2::PointD m_center;
Boundaries m_boundaries;
uint64_t m_population;
};
@ -52,16 +58,18 @@ public:
template <typename Fn>
bool WithBestLocality(Fn && fn) const
{
if (!m_bestLocality)
if (!m_locality)
return false;
fn(*m_bestLocality);
fn(*m_locality);
return true;
}
private:
m2::PointD const m_p;
double m_bestScore = std::numeric_limits<double>::max();
LocalityItem const * m_bestLocality = nullptr;
bool m_inside = false;
double m_score = std::numeric_limits<double>::max();
LocalityItem const * m_locality = nullptr;
};
class LocalityFinder
@ -91,7 +99,8 @@ public:
DISALLOW_COPY_AND_MOVE(Holder);
};
LocalityFinder(Index const & index, VillagesCache & villagesCache);
LocalityFinder(Index const & index, CitiesBoundariesTable const & boundaries,
VillagesCache & villagesCache);
template <typename Fn>
bool GetLocality(m2::PointD const & p, Fn && fn)
@ -116,6 +125,7 @@ private:
void UpdateMaps();
Index const & m_index;
CitiesBoundariesTable const & m_boundariesTable;
VillagesCache & m_villagesCache;
Holder m_cities;

View file

@ -183,8 +183,9 @@ Processor::Processor(Index const & index, CategoriesHolder const & categories,
, m_suggestsEnabled(true)
, m_viewportSearch(false)
, m_villagesCache(static_cast<my::Cancellable const &>(*this))
, m_ranker(index, infoGetter, m_emitter, categories, suggests, m_villagesCache,
static_cast<my::Cancellable const &>(*this))
, m_citiesBoundaries(index)
, m_ranker(index, m_citiesBoundaries, infoGetter, m_emitter, categories, suggests,
m_villagesCache, static_cast<my::Cancellable const &>(*this))
, m_preRanker(index, m_ranker, kPreResultsCount)
, m_geocoder(index, infoGetter, m_preRanker, m_villagesCache,
static_cast<my::Cancellable const &>(*this))
@ -390,6 +391,8 @@ void Processor::SetViewportByIndex(m2::RectD const & viewport, size_t idx, bool
void Processor::ClearCache(size_t ind) { m_viewport[ind].MakeEmpty(); }
void Processor::LoadCitiesBoundaries() { m_citiesBoundaries.Load(); }
Locales Processor::GetCategoryLocales() const
{
static int8_t const enLocaleCode = CategoriesHolder::MapLocaleToInteger("en");

View file

@ -1,6 +1,7 @@
#pragma once
#include "search/categories_cache.hpp"
#include "search/categories_set.hpp"
#include "search/cities_boundaries_table.hpp"
#include "search/emitter.hpp"
#include "search/geocoder.hpp"
#include "search/hotels_filter.hpp"
@ -111,6 +112,7 @@ public:
void InitEmitter();
void ClearCaches();
void LoadCitiesBoundaries();
protected:
enum ViewportID
@ -180,6 +182,7 @@ protected:
bool m_viewportSearch;
VillagesCache m_villagesCache;
CitiesBoundariesTable m_citiesBoundaries;
Emitter m_emitter;
Ranker m_ranker;

View file

@ -319,13 +319,13 @@ public:
// static
size_t const Ranker::kBatchSize = 10;
Ranker::Ranker(Index const & index, storage::CountryInfoGetter const & infoGetter,
Emitter & emitter, CategoriesHolder const & categories,
vector<Suggest> const & suggests, VillagesCache & villagesCache,
my::Cancellable const & cancellable)
Ranker::Ranker(Index const & index, CitiesBoundariesTable const & boundariesTable,
storage::CountryInfoGetter const & infoGetter, Emitter & emitter,
CategoriesHolder const & categories, vector<Suggest> const & suggests,
VillagesCache & villagesCache, my::Cancellable const & cancellable)
: m_reverseGeocoder(index)
, m_cancellable(cancellable)
, m_localities(index, villagesCache)
, m_localities(index, boundariesTable, villagesCache)
, m_index(index)
, m_infoGetter(infoGetter)
, m_emitter(emitter)

View file

@ -35,9 +35,10 @@ class CountryInfoGetter;
namespace search
{
class VillagesCache;
class CitiesBoundariesTable;
class Emitter;
class PreResult2Maker;
class VillagesCache;
class Ranker
{
@ -71,7 +72,8 @@ public:
static size_t const kBatchSize;
Ranker(Index const & index, storage::CountryInfoGetter const & infoGetter, Emitter & emitter,
Ranker(Index const & index, CitiesBoundariesTable const & boundariesTable,
storage::CountryInfoGetter const & infoGetter, Emitter & emitter,
CategoriesHolder const & categories, vector<Suggest> const & suggests,
VillagesCache & villagesCache, my::Cancellable const & cancellable);
virtual ~Ranker() = default;

View file

@ -79,6 +79,11 @@ bool SearchTest::ResultsMatch(SearchParams const & params, TRules const & rules)
return ResultsMatch(request.Results(), rules);
}
bool SearchTest::ResultMatches(search::Result const & result, TRule const & rule)
{
return tests_support::ResultMatches(m_engine, rule, result);
}
unique_ptr<tests_support::TestSearchRequest> SearchTest::MakeRequest(
string const & query, string const & locale /* = "en" */)
{

View file

@ -39,7 +39,8 @@ public:
class SearchTest : public TestWithClassificator
{
public:
using TRules = vector<shared_ptr<tests_support::MatchingRule>>;
using TRule = shared_ptr<tests_support::MatchingRule>;
using TRules = vector<TRule>;
SearchTest();
@ -83,7 +84,9 @@ public:
template <typename TBuildFn>
MwmSet::MwmId BuildWorld(TBuildFn && fn)
{
return BuildMwm("testWorld", feature::DataHeader::world, forward<TBuildFn>(fn));
auto const id = BuildMwm("testWorld", feature::DataHeader::world, forward<TBuildFn>(fn));
m_engine.LoadCitiesBoundaries();
return id;
}
template <typename TBuildFn>
@ -113,6 +116,8 @@ public:
bool ResultsMatch(SearchParams const & params, TRules const & rules);
bool ResultMatches(search::Result const & result, TRule const & rule);
unique_ptr<tests_support::TestSearchRequest> MakeRequest(string const & query,
string const & locale = "en");

View file

@ -1,6 +1,7 @@
#include "testing/testing.hpp"
#include "search/categories_cache.hpp"
#include "search/cities_boundaries_table.hpp"
#include "search/emitter.hpp"
#include "search/intermediate_result.hpp"
#include "search/model.hpp"
@ -44,11 +45,11 @@ namespace
class TestRanker : public Ranker
{
public:
TestRanker(TestSearchEngine & engine, Emitter & emitter, vector<Suggest> const & suggests,
VillagesCache & villagesCache, my::Cancellable const & cancellable,
vector<PreResult1> & results)
: Ranker(static_cast<Index const &>(engine), engine.GetCountryInfoGetter(), emitter,
GetDefaultCategories(), suggests, villagesCache, cancellable)
TestRanker(TestSearchEngine & engine, CitiesBoundariesTable const & boundariesTable,
Emitter & emitter, vector<Suggest> const & suggests, VillagesCache & villagesCache,
my::Cancellable const & cancellable, vector<PreResult1> & results)
: Ranker(static_cast<Index const &>(engine), boundariesTable, engine.GetCountryInfoGetter(),
emitter, GetDefaultCategories(), suggests, villagesCache, cancellable)
, m_results(results)
{
}
@ -113,8 +114,10 @@ UNIT_CLASS_TEST(PreRankerTest, Smoke)
vector<PreResult1> results;
Emitter emitter;
CitiesBoundariesTable boundariesTable(m_engine);
VillagesCache villagesCache(m_cancellable);
TestRanker ranker(m_engine, emitter, m_suggests, villagesCache, m_cancellable, results);
TestRanker ranker(m_engine, boundariesTable, emitter, m_suggests, villagesCache, m_cancellable,
results);
PreRanker preRanker(m_engine, ranker, pois.size());
PreRanker::Params params;

View file

@ -1113,8 +1113,8 @@ UNIT_CLASS_TEST(ProcessorTest, CityBoundaryLoad)
TEST(ResultsMatch("moscow", "en", rules), ());
}
CitiesBoundariesTable table;
TEST(table.Load(m_engine), ());
CitiesBoundariesTable table(m_engine);
TEST(table.Load(), ());
TEST(table.Has(0 /* fid */), ());
TEST(!table.Has(10 /* fid */), ());
@ -1129,5 +1129,54 @@ UNIT_CLASS_TEST(ProcessorTest, CityBoundaryLoad)
TEST(!boundaries.HasPoint(m2::PointD(0.6, 0.6)), ());
TEST(!boundaries.HasPoint(m2::PointD(-1, 0.5)), ());
}
UNIT_CLASS_TEST(ProcessorTest, CityBoundarySmoke)
{
TestCity moscow(vector<m2::PointD>({m2::PointD(0, 0), m2::PointD(0.5, 0), m2::PointD(0.5, 0.5),
m2::PointD(0, 0.5)}),
"Москва", "ru", 100 /* rank */);
TestCity khimki(vector<m2::PointD>({m2::PointD(0.25, 0.5), m2::PointD(0.5, 0.5),
m2::PointD(0.5, 0.75), m2::PointD(0.25, 0.75)}),
"Химки", "ru", 50 /* rank */);
TestPOI cafeMoscow(m2::PointD(0.49, 0.49), "Москвичка", "ru");
cafeMoscow.SetTypes({{"amenity", "cafe"}, {"internet_access", "wlan"}});
TestPOI cafeKhimki(m2::PointD(0.49, 0.51), "Химичка", "ru");
cafeKhimki.SetTypes({{"amenity", "cafe"}, {"internet_access", "wlan"}});
BuildWorld([&](TestMwmBuilder & builder) {
builder.Add(moscow);
builder.Add(khimki);
});
auto countryId = BuildCountry("Россия" /* countryName */, [&](TestMwmBuilder & builder) {
builder.Add(cafeMoscow);
builder.Add(cafeKhimki);
});
SetViewport(m2::RectD(m2::PointD(-1.0, -1.0), m2::PointD(1.0, 1.0)));
{
auto request = MakeRequest("кафе", "ru");
auto const & results = request->Results();
TRules rules{ExactMatch(countryId, cafeMoscow), ExactMatch(countryId, cafeKhimki)};
TEST(ResultsMatch(results, rules), ());
for (auto const & result : results)
{
if (ResultMatches(result, ExactMatch(countryId, cafeMoscow)))
{
TEST_EQUAL(result.GetAddress(), "Москва, Россия", ());
}
else
{
TEST(ResultMatches(result, ExactMatch(countryId, cafeKhimki)), ());
TEST_EQUAL(result.GetAddress(), "Химки, Россия", ());
}
}
}
}
} // namespace
} // namespace search

View file

@ -29,12 +29,16 @@ class LocalityFinderTest : public TestWithClassificator
my::Cancellable m_cancellable;
search::VillagesCache m_villagesCache;
search::CitiesBoundariesTable m_boundariesTable;
search::LocalityFinder m_finder;
m2::RectD m_worldRect;
public:
LocalityFinderTest() : m_villagesCache(m_cancellable), m_finder(m_index, m_villagesCache)
LocalityFinderTest()
: m_villagesCache(m_cancellable)
, m_boundariesTable(m_index)
, m_finder(m_index, m_boundariesTable, m_villagesCache)
{
m_worldFile = platform::LocalCountryFile::MakeForTesting("World");
@ -47,6 +51,7 @@ public:
TEST(id.IsAlive(), ());
m_worldRect = id.GetInfo()->m_limitRect;
m_boundariesTable.Load();
}
catch (RootException const & ex)
{

View file

@ -18,7 +18,7 @@ StringUtf8Multilang ToMultilang(string const & name)
struct City
{
City(string const & name, m2::PointD const & center, uint64_t population)
: m_item(ToMultilang(name), center, population)
: m_item(ToMultilang(name), center, {} /* boundaries */, population)
{
}
@ -37,18 +37,14 @@ string GetMatchedCity(m2::PointD const & point, vector<City> const & cities)
return name;
}
// TODO (@y): this test fails for now. Need to uncomment it as soon as
// locality finder will be fixed.
//
// UNIT_TEST(LocalitySelector_Test1)
// {
// auto const name = GetMatchedCity(
// m2::PointD(-97.56345, 26.79672),
// {{"Matamoros", m2::PointD(-97.50665, 26.79718), 10000},
UNIT_TEST(LocalitySelector_Test1)
{
auto const name = GetMatchedCity(m2::PointD(-97.56345, 26.79672),
{{"Matamoros", m2::PointD(-97.50665, 26.79718), 918536},
// {"Brownsville", m2::PointD(-97.48910, 26.84558), 180663}});
// TEST_EQUAL(name, "Matamoros", ());
// }
{"Brownsville", m2::PointD(-97.48910, 26.84558), 180663}});
TEST_EQUAL(name, "Matamoros", ());
}
UNIT_TEST(LocalitySelector_Test2)
{

View file

@ -99,6 +99,14 @@ bool MatchResults(Index const & index, vector<shared_ptr<MatchingRule>> rules,
return false;
}
bool ResultMatches(Index const & index, shared_ptr<MatchingRule> rule,
search::Result const & result)
{
bool matches = false;
index.ReadFeature([&](FeatureType & ft) { matches = rule->Matches(ft); }, result.GetFeatureID());
return matches;
}
string DebugPrint(MatchingRule const & rule) { return rule.ToString(); }
} // namespace tests_support
} // namespace search

View file

@ -74,6 +74,8 @@ shared_ptr<MatchingRule> AlternativesMatch(TArgs &&... args)
bool MatchResults(Index const & index, vector<shared_ptr<MatchingRule>> rules,
vector<search::Result> const & actual);
bool ResultMatches(Index const & index, shared_ptr<MatchingRule> rule,
search::Result const & result);
string DebugPrint(MatchingRule const & rule);
} // namespace tests_support

View file

@ -31,7 +31,9 @@ public:
TestSearchEngine(unique_ptr<::search::ProcessorFactory> factory, Engine::Params const & params);
~TestSearchEngine() override;
inline void SetLocale(string const & locale) { m_engine.SetLocale(locale); }
void SetLocale(string const & locale) { m_engine.SetLocale(locale); }
void LoadCitiesBoundaries() { m_engine.LoadCitiesBoundariesForTesting(); }
weak_ptr<search::ProcessorHandle> Search(search::SearchParams const & params,
m2::RectD const & viewport);