[search] Fixed features sorting for viewport search.

This commit is contained in:
Yuri Gorshenin 2016-04-05 18:12:57 +03:00
parent 825c0716c8
commit 8b0d48f745
7 changed files with 72 additions and 16 deletions

View file

@ -115,6 +115,7 @@ public:
inline FeatureID const & GetID() const { return m_id; }
inline string const & GetName() const { return m_str; }
inline feature::TypesHolder const & GetTypes() const { return m_types; }
inline m2::PointD GetCenter() const { return m_region.m_point; }
private:
bool IsEqualCommon(PreResult2 const & r) const;
@ -143,8 +144,6 @@ private:
string GetRegionName(storage::CountryInfoGetter const & infoGetter, uint32_t fType) const;
m2::PointD GetCenter() const { return m_region.m_point; }
double m_distance;
ResultType m_resultType;
v2::RankingInfo m_info;

View file

@ -23,6 +23,14 @@ namespace search
{
namespace
{
void MakeDefaultTestParams(string const & query, SearchParams & params)
{
params.m_query = query;
params.m_inputLocale = "en";
params.SetMode(Mode::Everywhere);
params.SetSuggestsEnabled(false);
}
class SearchQueryV2Test : public SearchTest
{
};
@ -267,7 +275,6 @@ UNIT_CLASS_TEST(SearchQueryV2Test, TestRankingInfo)
string const countryName = "Wonderland";
TestCity sanFrancisco(m2::PointD(1, 1), "San Francisco", "en", 100 /* rank */);
// Golden Gate Bridge-bridge is located in this test on the Golden
// Gate Bridge-street. Therefore, there are several valid parses of
// the query "Golden Gate Bridge", and search engine must return
@ -277,24 +284,35 @@ UNIT_CLASS_TEST(SearchQueryV2Test, TestRankingInfo)
vector<m2::PointD>{m2::PointD(-0.5, -0.5), m2::PointD(0, 0), m2::PointD(0.5, 0.5)},
"Golden Gate Bridge", "en");
TestPOI goldenGateBridge(m2::PointD(0, 0), "Golden Gate Bridge", "en");
TestPOI lermontov(m2::PointD(1, 1), "Лермонтовъ", "en");
lermontov.SetTypes({{"amenity", "cafe"}});
// A city with two noname cafes.
TestCity lermontovo(m2::PointD(-1, -1), "Лермонтово", "en", 100 /* rank */);
TestPOI cafe1(m2::PointD(-1.01, -1.01), "", "en");
cafe1.SetTypes({{"amenity", "cafe"}});
TestPOI cafe2(m2::PointD(-0.99, -0.99), "", "en");
cafe2.SetTypes({{"amenity", "cafe"}});
auto worldId = BuildMwm("testWorld", feature::DataHeader::world, [&](TestMwmBuilder & builder)
{
builder.Add(sanFrancisco);
builder.Add(lermontovo);
});
auto wonderlandId = BuildMwm(countryName, feature::DataHeader::country, [&](TestMwmBuilder & builder)
{
builder.Add(goldenGateStreet);
builder.Add(cafe1);
builder.Add(cafe2);
builder.Add(goldenGateBridge);
builder.Add(goldenGateStreet);
builder.Add(lermontov);
});
SetViewport(m2::RectD(m2::PointD(-0.5, -0.5), m2::PointD(0.5, 0.5)));
{
SearchParams params;
params.m_query = "golden gate bridge ";
params.m_inputLocale = "en";
params.SetMode(Mode::Everywhere);
params.SetSuggestsEnabled(false);
MakeDefaultTestParams("golden gate bridge ", params);
TestSearchRequest request(m_engine, params, m_viewport);
request.Wait();
@ -310,6 +328,25 @@ UNIT_CLASS_TEST(SearchQueryV2Test, TestRankingInfo)
TEST(my::AlmostEqualAbs(1.0, info.m_nameCoverage, 1e-6), (info.m_nameCoverage));
}
}
// This test is quite important and must always pass.
{
SearchParams params;
MakeDefaultTestParams("cafe лермонтов", params);
TestSearchRequest request(m_engine, params, m_viewport);
request.Wait();
auto const & results = request.Results();
TRules rules{ExactMatch(wonderlandId, cafe1), ExactMatch(wonderlandId, cafe2),
ExactMatch(wonderlandId, lermontov)};
TEST(MatchResults(m_engine, rules, results), ());
TEST_EQUAL(3, results.size(), ("Unexpected number of retrieved cafes."));
auto const & top = results.front();
TEST(MatchResults(m_engine, {ExactMatch(wonderlandId, lermontov)}, {top}), ());
}
}
} // namespace
} // namespace search

View file

@ -125,16 +125,25 @@ class IndexedValue : public search::IndexedValueBase<Query::kQueuesCount>
shared_ptr<impl::PreResult2> m_val;
double m_rank;
double m_distanceToPivot;
public:
explicit IndexedValue(unique_ptr<impl::PreResult2> v)
: m_val(move(v)), m_rank(m_val ? m_val->GetRankingInfo().GetLinearModelRank() : 0)
: m_val(move(v)), m_rank(0), m_distanceToPivot(numeric_limits<double>::max())
{
if (!m_val)
return;
auto const & info = m_val->GetRankingInfo();
m_rank = info.GetLinearModelRank();
m_distanceToPivot = info.m_distanceToPivot;
}
impl::PreResult2 const & operator*() const { return *m_val; }
inline double GetRank() const { return m_rank; }
inline double GetDistanceToPivot() const { return m_distanceToPivot; }
};
string DebugPrint(IndexedValue const & value)
@ -523,15 +532,17 @@ void Query::FlushViewportResults(v2::Geocoder::Params const & params, Results &
vector<FeatureID> streets;
MakePreResult2(params, indV, streets);
RemoveDuplicatingLinear(indV);
if (indV.empty())
return;
RemoveDuplicatingLinear(indV);
sort(indV.begin(), indV.end(), my::CompareBy(&IndexedValue::GetDistanceToPivot));
for (size_t i = 0; i < indV.size(); ++i)
{
if (IsCancelled())
break;
res.AddResultNoChecks((*(indV[i])).GenerateFinalResult(m_infoGetter, &m_categories,
&m_prefferedTypes, m_currentLocaleCode, m_reverseGeocoder));
}
@ -781,6 +792,9 @@ void Query::MakePreResult2(v2::Geocoder::Params const & params, vector<T> & cont
if (!p)
continue;
if (params.m_mode == Mode::Viewport && !params.m_pivot.IsPointInside(p->GetCenter()))
continue;
if (p->IsStreet())
streets.push_back(p->GetID());
@ -796,12 +810,10 @@ void Query::FlushResults(v2::Geocoder::Params const & params, Results & res, boo
vector<FeatureID> streets;
MakePreResult2(params, indV, streets);
RemoveDuplicatingLinear(indV);
if (indV.empty())
return;
RemoveDuplicatingLinear(indV);
sort(indV.rbegin(), indV.rend(), my::CompareBy(&IndexedValue::GetRank));
// Do not process suggestions in additional search.

View file

@ -52,7 +52,8 @@ void TestFeature::Serialize(FeatureBuilder1 & fb) const
fb.SetTestId(m_id);
if (m_hasCenter)
fb.SetCenter(m_center);
CHECK(fb.AddName(m_lang, m_name), ("Can't set feature name:", m_name, "(", m_lang, ")"));
if (!m_name.empty())
CHECK(fb.AddName(m_lang, m_name), ("Can't set feature name:", m_name, "(", m_lang, ")"));
}
// TestCountry -------------------------------------------------------------------------------------
@ -155,13 +156,17 @@ string TestStreet::ToString() const
TestPOI::TestPOI(m2::PointD const & center, string const & name, string const & lang)
: TestFeature(center, name, lang)
{
m_types = {{"railway", "station"}};
}
void TestPOI::Serialize(FeatureBuilder1 & fb) const
{
TestFeature::Serialize(fb);
auto const & classificator = classif();
fb.SetType(classificator.GetTypeByPath({"railway", "station"}));
for (auto const & path : m_types)
fb.SetType(classificator.GetTypeByPath(path));
if (!m_houseNumber.empty())
fb.AddHouseNumber(m_houseNumber);
if (!m_streetName.empty())

View file

@ -94,10 +94,12 @@ public:
inline void SetHouseNumber(string const & houseNumber) { m_houseNumber = houseNumber; }
inline void SetStreet(TestStreet const & street) { m_streetName = street.GetName(); }
inline void SetTypes(vector<vector<string>> const & types) { m_types = types; }
private:
string m_houseNumber;
string m_streetName;
vector<vector<string>> m_types;
};
class TestBuilding : public TestFeature

View file

@ -548,7 +548,7 @@ void Geocoder::GoImpl(vector<shared_ptr<MwmInfo>> & infos, bool inViewport)
// found.
size_t const numIntersectingMaps = OrderCountries(m_params.m_pivot, infos);
// MatchViewportAndPosition() should always be matched in mwms
// MatchAroundPivot() should always be matched in mwms
// intersecting with position and viewport.
auto const & cancellable = static_cast<my::Cancellable const&>(*this);
auto processCountry = [&](size_t index, unique_ptr<MwmContext> context)

View file

@ -59,6 +59,7 @@ void SearchQueryV2::SearchViewportPoints(Results & res)
Geocoder::Params params;
InitParams(false /* localitySearch */, params);
params.m_pivot = m_viewport[CURRENT_V];
params.m_accuratePivotCenter = params.m_pivot.Center();
params.m_maxNumResults = kPreResultsCount;
m_geocoder.SetParams(params);