[search] Limited ReverseGeocoder with single MWM while matching streets.

This commit is contained in:
vng 2016-01-28 16:39:40 +03:00 committed by Sergey Yershov
parent 41cef46ddc
commit d73fc8b5e4
11 changed files with 135 additions and 92 deletions

View file

@ -318,7 +318,8 @@ void BuildAddressTable(FilesContainerR & container, Writer & writer)
Index mwmIndex;
/// @ todo Make some better solution, or legalize MakeTemporary.
mwmIndex.RegisterMap(platform::LocalCountryFile::MakeTemporary(container.GetFileName()));
auto const mwmId = mwmIndex.RegisterMap(platform::LocalCountryFile::MakeTemporary(container.GetFileName()));
ASSERT_EQUAL(mwmId.second, MwmSet::RegResult::Success, ());
search::ReverseGeocoder rgc(mwmIndex);
{
@ -338,10 +339,11 @@ void BuildAddressTable(FilesContainerR & container, Writer & writer)
{
FeatureType ft;
features.GetVector().GetByIndex(index, ft);
ft.SetID(FeatureID(mwmId.first, index));
using TStreet = search::ReverseGeocoder::Street;
vector<TStreet> streets;
rgc.GetNearbyStreets(feature::GetCenter(ft), streets);
rgc.GetNearbyStreets(ft, streets);
streetIndex = rgc.GetMatchedStreetIndex(street, streets);
if (streetIndex < streets.size())

View file

@ -2,6 +2,7 @@
#include "search_string_utils.hpp"
#include "search/v2/house_to_street_table.hpp"
#include "search/v2/mwm_context.hpp"
#include "indexer/feature.hpp"
#include "indexer/feature_algo.hpp"
@ -26,17 +27,18 @@ double const ReverseGeocoder::kLookupRadiusM = 500.0;
ReverseGeocoder::ReverseGeocoder(Index const & index) : m_index(index) {}
void ReverseGeocoder::GetNearbyStreets(m2::PointD const & center, vector<Street> & streets) const
void ReverseGeocoder::GetNearbyStreets(MwmSet::MwmId id, m2::PointD const & center,
vector<Street> & streets) const
{
m2::RectD const rect = GetLookupRect(center, kLookupRadiusM);
auto const addStreet = [&](FeatureType const & ft)
auto const addStreet = [&](FeatureType & ft)
{
if (ft.GetFeatureType() != feature::GEOM_LINE)
return;
if (!ftypes::IsStreetChecker::Instance()(ft))
if (ft.GetFeatureType() != feature::GEOM_LINE ||
!ftypes::IsStreetChecker::Instance()(ft))
{
return;
}
string name;
static int8_t const lang = StringUtf8Multilang::GetLangIndex("default");
@ -48,8 +50,18 @@ void ReverseGeocoder::GetNearbyStreets(m2::PointD const & center, vector<Street>
streets.push_back({ft.GetID(), feature::GetMinDistanceMeters(ft, center), name});
};
m_index.ForEachInRect(addStreet, rect, kQueryScale);
sort(streets.begin(), streets.end(), my::CompareBy(&Street::m_distanceMeters));
MwmSet::MwmHandle mwmHandle = m_index.GetMwmHandleById(id);
if (mwmHandle.IsAlive())
{
search::v2::MwmContext(move(mwmHandle)).ForEachFeature(rect, addStreet);
sort(streets.begin(), streets.end(), my::CompareBy(&Street::m_distanceMeters));
}
}
void ReverseGeocoder::GetNearbyStreets(FeatureType & ft, vector<Street> & streets) const
{
ASSERT(ft.GetID().IsValid(), ());
GetNearbyStreets(ft.GetID().m_mwmId, feature::GetCenter(ft), streets);
}
// static
@ -108,7 +120,7 @@ void ReverseGeocoder::GetNearbyAddress(m2::PointD const & center, Address & addr
table = search::v2::HouseToStreetTable::Load(*mwmHandle.GetValue<MwmValue>());
}
GetNearbyStreets(b.m_center, streets);
GetNearbyStreets(b.m_id.m_mwmId, b.m_center, streets);
uint32_t ind;
if (table->Get(b.m_id.m_index, ind) && ind < streets.size())
@ -120,8 +132,8 @@ void ReverseGeocoder::GetNearbyAddress(m2::PointD const & center, Address & addr
}
}
pair<vector<ReverseGeocoder::Street>, uint32_t> ReverseGeocoder::GetNearbyFeatureStreets(
FeatureType const & feature) const
pair<vector<ReverseGeocoder::Street>, uint32_t>
ReverseGeocoder::GetNearbyFeatureStreets(FeatureType const & feature) const
{
pair<vector<ReverseGeocoder::Street>, uint32_t> result;
auto & streetIndex = result.second;
@ -131,33 +143,24 @@ pair<vector<ReverseGeocoder::Street>, uint32_t> ReverseGeocoder::GetNearbyFeatur
MwmSet::MwmHandle const mwmHandle = m_index.GetMwmHandleById(fid.m_mwmId);
if (!mwmHandle.IsAlive())
{
LOG(LERROR, ("Feature handle is not alive", feature));
LOG(LWARNING, ("MWM for", feature, "is dead"));
return result;
}
auto & streets = result.first;
GetNearbyStreets(feature::GetCenter(feature), streets);
GetNearbyStreets(const_cast<FeatureType &>(feature), streets);
unique_ptr<search::v2::HouseToStreetTable> const table =
search::v2::HouseToStreetTable::Load(*mwmHandle.GetValue<MwmValue>());
if (table->Get(fid.m_index, streetIndex) && streetIndex >= streets.size())
LOG(LERROR, ("Critical reverse geocoder error, returned", streetIndex, "for", feature));
LOG(LWARNING, ("Out of bound index", streetIndex, "for", feature));
return result;
}
void ReverseGeocoder::GetNearbyBuildings(m2::PointD const & center, vector<Building> & buildings) const
{
GetNearbyBuildings(center, kLookupRadiusM, buildings);
}
void ReverseGeocoder::GetNearbyBuildings(m2::PointD const & center, double radiusM,
vector<Building> & buildings) const
{
// Seems like a copy-paste here of the GetNearbyStreets function.
// Trying to factor out common logic will cause many variables logic.
m2::RectD const rect = GetLookupRect(center, radiusM);
m2::RectD const rect = GetLookupRect(center, kLookupRadiusM);
auto const addBuilding = [&](FeatureType const & ft)
{
@ -169,8 +172,8 @@ void ReverseGeocoder::GetNearbyBuildings(m2::PointD const & center, double radiu
if (number.empty())
return;
buildings.push_back(
{ft.GetID(), feature::GetMinDistanceMeters(ft, center), number, feature::GetCenter(ft)});
buildings.push_back({ft.GetID(), feature::GetMinDistanceMeters(ft, center),
number, feature::GetCenter(ft)});
};
m_index.ForEachInRect(addBuilding, rect, kQueryScale);
@ -182,4 +185,5 @@ m2::RectD ReverseGeocoder::GetLookupRect(m2::PointD const & center, double radiu
{
return MercatorBounds::RectByCenterXYAndSizeInMeters(center, radiusM);
}
} // namespace search

View file

@ -33,6 +33,7 @@ class ReverseGeocoder
};
public:
/// All "Nearby" functions work in this lookup radius.
static double const kLookupRadiusM;
explicit ReverseGeocoder(Index const & index);
@ -65,17 +66,22 @@ public:
double GetDistance() const { return m_building.m_distanceMeters; }
};
void GetNearbyStreets(m2::PointD const & center, vector<Street> & streets) const;
/// @return Sorted by distance streets vector for the specified MwmId.
//@{
void GetNearbyStreets(MwmSet::MwmId id, m2::PointD const & center,
vector<Street> & streets) const;
void GetNearbyStreets(FeatureType & ft, vector<Street> & streets) const;
//@}
void GetNearbyAddress(m2::PointD const & center, Address & addr) const;
/// @returns street segments (can be duplicate names) sorted by distance to feature's center.
/// uint32_t, if less than vector.size(), contains index of exact feature's street specified in OSM data.
/// @todo Leave const reference for now to support client's legacy code.
/// It's better to use honest non-const reference when feature can be modified in any way.
pair<vector<Street>, uint32_t> GetNearbyFeatureStreets(FeatureType const & feature) const;
void GetNearbyBuildings(m2::PointD const & center, vector<Building> & buildings) const;
/// @return The nearest exact address where building has house number and valid street match.
void GetNearbyAddress(m2::PointD const & center, Address & addr) const;
void GetNearbyBuildings(m2::PointD const & center, double radiusM, vector<Building> & buildings) const;
/// @return Sorted by distance houses vector with valid house number.
void GetNearbyBuildings(m2::PointD const & center, vector<Building> & buildings) const;
private:
static m2::RectD GetLookupRect(m2::PointD const & center, double radiusM);

View file

@ -65,7 +65,8 @@ uint32_t FeaturesLayerMatcher::GetMatchingStreet(uint32_t houseId, FeatureType &
return entry.first;
}
vector<ReverseGeocoder::Street> const & FeaturesLayerMatcher::GetNearbyStreets(uint32_t featureId)
vector<FeaturesLayerMatcher::TStreet> const &
FeaturesLayerMatcher::GetNearbyStreets(uint32_t featureId)
{
auto entry = m_nearbyStreetsCache.Get(featureId);
if (!entry.second)
@ -74,21 +75,34 @@ vector<ReverseGeocoder::Street> const & FeaturesLayerMatcher::GetNearbyStreets(u
FeatureType feature;
GetByIndex(featureId, feature);
m_reverseGeocoder.GetNearbyStreets(feature::GetCenter(feature), entry.first);
GetNearbyStreetsImpl(feature, entry.first);
return entry.first;
}
vector<ReverseGeocoder::Street> const & FeaturesLayerMatcher::GetNearbyStreets(
uint32_t featureId, FeatureType & feature)
vector<FeaturesLayerMatcher::TStreet> const &
FeaturesLayerMatcher::GetNearbyStreets(uint32_t featureId, FeatureType & feature)
{
auto entry = m_nearbyStreetsCache.Get(featureId);
if (!entry.second)
return entry.first;
m_reverseGeocoder.GetNearbyStreets(feature::GetCenter(feature), entry.first);
GetNearbyStreetsImpl(feature, entry.first);
return entry.first;
}
void FeaturesLayerMatcher::GetNearbyStreetsImpl(FeatureType & feature, vector<TStreet> & streets)
{
m_reverseGeocoder.GetNearbyStreets(feature, streets);
for (size_t i = 0; i < streets.size(); ++i)
{
if (streets[i].m_distanceMeters > ReverseGeocoder::kLookupRadiusM)
{
streets.resize(i);
return;
}
}
}
uint32_t FeaturesLayerMatcher::GetMatchingStreetImpl(uint32_t houseId, FeatureType & houseFeature)
{
auto const & streets = GetNearbyStreets(houseId, houseFeature);

View file

@ -158,21 +158,19 @@ private:
if (queryTokens.empty())
return;
vector<ReverseGeocoder::Building> nearbyBuildings;
for (size_t i = 0; i < pois.size(); ++i)
{
// TODO (@y, @m, @vng): implement a faster version
// ReverseGeocoder::GetNearbyBuildings() for only one (current)
// map.
nearbyBuildings.clear();
m_reverseGeocoder.GetNearbyBuildings(poiCenters[i], kBuildingRadiusMeters, nearbyBuildings);
for (auto const & building : nearbyBuildings)
{
if (building.m_id.m_mwmId != m_context->m_id || building.m_distanceMeters > kBuildingRadiusMeters)
continue;
if (HouseNumbersMatch(strings::MakeUniString(building.m_name), queryTokens))
fn(pois[i], building.m_id.m_index);
}
m_context->ForEachFeature(
MercatorBounds::RectByCenterXYAndSizeInMeters(poiCenters[i], kBuildingRadiusMeters),
[&](FeatureType & ft)
{
if (HouseNumbersMatch(strings::MakeUniString(ft.GetHouseNumber()), queryTokens))
{
double const distanceM = MercatorBounds::DistanceOnEarth(feature::GetCenter(ft), poiCenters[i]);
if (distanceM < kBuildingRadiusMeters)
fn(pois[i], ft.GetID().m_index);
}
});
}
}
@ -191,17 +189,8 @@ private:
{
for (uint32_t poiId : pois)
{
// TODO (@y, @m, @vng): implement a faster version
// ReverseGeocoder::GetNearbyStreets() for only one (current)
// map.
for (auto const & street : GetNearbyStreets(poiId))
{
if (street.m_id.m_mwmId != m_context->m_id ||
street.m_distanceMeters > ReverseGeocoder::kLookupRadiusM)
{
continue;
}
uint32_t const streetId = street.m_id.m_index;
if (binary_search(streets.begin(), streets.end(), streetId))
fn(poiId, streetId);
@ -331,20 +320,20 @@ private:
// Returns id of a street feature corresponding to a |houseId|, or
// kInvalidId if there're not such street.
uint32_t GetMatchingStreet(uint32_t houseId);
uint32_t GetMatchingStreet(uint32_t houseId, FeatureType & houseFeature);
vector<ReverseGeocoder::Street> const & GetNearbyStreets(uint32_t featureId);
vector<ReverseGeocoder::Street> const & GetNearbyStreets(uint32_t featureId,
FeatureType & feature);
uint32_t GetMatchingStreetImpl(uint32_t houseId, FeatureType & houseFeature);
using TStreet = ReverseGeocoder::Street;
vector<TStreet> const & GetNearbyStreets(uint32_t featureId);
vector<TStreet> const & GetNearbyStreets(uint32_t featureId,
FeatureType & feature);
void GetNearbyStreetsImpl(FeatureType & feature, vector<TStreet> & streets);
inline void GetByIndex(uint32_t id, FeatureType & ft) const
{
/// @todo Add Cache for feature id -> (point, name / house number).
m_context->m_vector.GetByIndex(id, ft);
m_context->GetFeature(id, ft);
}
MwmContext * m_context;
@ -353,7 +342,7 @@ private:
// Cache of streets in a feature's vicinity. All lists in the cache
// are ordered by distance from the corresponding feature.
Cache<uint32_t, vector<ReverseGeocoder::Street>> m_nearbyStreetsCache;
Cache<uint32_t, vector<TStreet>> m_nearbyStreetsCache;
// Cache of correct streets for buildings. Current search algorithm
// supports only one street for a building, whereas buildings can be

View file

@ -527,10 +527,10 @@ void Geocoder::GoImpl(vector<shared_ptr<MwmInfo>> & infos, bool inViewport)
m_villages = nullptr;
});
auto it = m_matchersCache.find(m_context->m_id);
auto it = m_matchersCache.find(m_context->GetId());
if (it == m_matchersCache.end())
{
it = m_matchersCache.insert(make_pair(m_context->m_id, make_unique<FeaturesLayerMatcher>(
it = m_matchersCache.insert(make_pair(m_context->GetId(), make_unique<FeaturesLayerMatcher>(
m_index, cancellable))).first;
}
m_matcher = it->second.get();
@ -540,7 +540,7 @@ void Geocoder::GoImpl(vector<shared_ptr<MwmInfo>> & infos, bool inViewport)
if (inViewport)
{
viewportCBV =
Retrieval::RetrieveGeometryFeatures(m_context->m_id, m_context->m_value, cancellable,
Retrieval::RetrieveGeometryFeatures(m_context->GetId(), m_context->m_value, cancellable,
m_params.m_viewport, m_params.m_scale);
}
@ -616,7 +616,7 @@ void Geocoder::PrepareAddressFeatures()
{
PrepareRetrievalParams(i, i + 1);
m_addressFeatures[i] = Retrieval::RetrieveAddressFeatures(
m_context->m_id, m_context->m_value, static_cast<my::Cancellable const &>(*this),
m_context->GetId(), m_context->m_value, static_cast<my::Cancellable const &>(*this),
m_retrievalParams);
ASSERT(m_addressFeatures[i], ());
}
@ -644,7 +644,7 @@ void Geocoder::FillLocalityCandidates(coding::CompressedBitVector const * filter
[&](uint32_t featureId)
{
Locality l;
l.m_countryId = m_context->m_id;
l.m_countryId = m_context->GetId();
l.m_featureId = featureId;
l.m_startToken = startToken;
l.m_endToken = endToken;
@ -675,7 +675,7 @@ void Geocoder::FillLocalitiesTable()
for (auto & l : preLocalities)
{
FeatureType ft;
m_context->m_vector.GetByIndex(l.m_featureId, ft);
m_context->GetFeature(l.m_featureId, ft);
switch (m_model.GetSearchType(ft))
{
@ -1220,7 +1220,7 @@ void Geocoder::FindPaths()
ASSERT(result.IsValid(), ());
// TODO(@y, @m, @vng): use rest fields of IntersectionResult for
// better scoring.
m_results->emplace_back(m_context->m_id, result.InnermostResult());
m_results->emplace_back(m_context->GetId(), result.InnermostResult());
});
}
@ -1277,7 +1277,7 @@ unique_ptr<coding::CompressedBitVector> Geocoder::LoadCategories(
{
m_retrievalParams.m_tokens[0][0] = category;
auto cbv = Retrieval::RetrieveAddressFeatures(
context.m_id, context.m_value, static_cast<my::Cancellable const &>(*this),
context.GetId(), context.m_value, static_cast<my::Cancellable const &>(*this),
m_retrievalParams);
if (!coding::CompressedBitVector::IsEmpty(cbv))
cbvs.push_back(move(cbv));
@ -1323,14 +1323,14 @@ coding::CompressedBitVector const * Geocoder::RetrieveGeometryFeatures(MwmContex
/// - Implement more smart strategy according to id.
/// - Move all rect limits here
auto & features = m_geometryFeatures[context.m_id];
auto & features = m_geometryFeatures[context.GetId()];
for (auto const & v : features)
{
if (v.m_rect.IsRectInside(rect))
return v.m_cbv.get();
}
auto cbv = Retrieval::RetrieveGeometryFeatures(context.m_id, context.m_value,
auto cbv = Retrieval::RetrieveGeometryFeatures(context.GetId(), context.m_value,
static_cast<my::Cancellable const &>(*this), rect,
m_params.m_scale);

View file

@ -14,7 +14,7 @@ class HouseToStreetTable
public:
virtual ~HouseToStreetTable() = default;
/// @todo Actually, value may be nullptr in the very common case .
/// @todo Actually, value may be nullptr in the very common case.
/// It's better to construct a table from MwmHandle.
static unique_ptr<HouseToStreetTable> Load(MwmValue & value);

View file

@ -1,6 +1,5 @@
#include "search/v2/mwm_context.hpp"
#include "indexer/index.hpp"
namespace search
{
@ -9,14 +8,11 @@ namespace v2
MwmContext::MwmContext(MwmSet::MwmHandle handle)
: m_handle(move(handle))
, m_value(*m_handle.GetValue<MwmValue>())
, m_id(m_handle.GetId())
, m_vector(m_value.m_cont, m_value.GetHeader(), m_value.m_table)
, m_index(m_value.m_cont.GetReader(INDEX_FILE_TAG), m_value.m_factory)
{
}
string const & MwmContext::GetName() const { return m_id.GetInfo()->GetCountryName(); }
shared_ptr<MwmInfo> const & MwmContext::GetInfo() const { return m_id.GetInfo(); }
shared_ptr<MwmInfo> const & MwmContext::GetInfo() const { return GetId().GetInfo(); }
} // namespace v2
} // namespace search

View file

@ -1,7 +1,7 @@
#pragma once
#include "indexer/features_vector.hpp"
#include "indexer/mwm_set.hpp"
#include "indexer/index.hpp"
#include "indexer/scale_index.hpp"
#include "base/macros.hpp"
@ -18,13 +18,42 @@ struct MwmContext
MwmSet::MwmHandle m_handle;
MwmValue & m_value;
MwmSet::MwmId const & m_id;
FeaturesVector m_vector;
ScaleIndex<ModelReaderPtr> m_index;
string const & GetName() const;
inline MwmSet::MwmId const & GetId() const { return m_handle.GetId(); }
inline string const & GetName() const { return GetInfo()->GetCountryName(); }
shared_ptr<MwmInfo> const & GetInfo() const;
template <class TFn> void ForEachFeature(m2::RectD const & rect, TFn && fn)
{
feature::DataHeader const & header = m_value.GetHeader();
covering::CoveringGetter covering(rect, covering::ViewportWithLowLevels);
uint32_t const scale = header.GetLastScale();
covering::IntervalsT const & interval = covering.Get(scale);
CheckUniqueIndexes checkUnique(header.GetFormat() >= version::Format::v5);
for (auto const & i : interval)
{
m_index.ForEachInIntervalAndScale([&](uint32_t index)
{
if (checkUnique(index))
{
FeatureType ft;
GetFeature(index, ft);
fn(ft);
}
}, i.first, i.second, scale);
}
}
inline void GetFeature(uint32_t index, FeatureType & ft) const
{
m_vector.GetByIndex(index, ft);
ft.SetID(FeatureID(GetId(), index));
}
DISALLOW_COPY_AND_MOVE(MwmContext);
};
} // namespace v2

View file

@ -49,7 +49,7 @@ StreetVicinityLoader::Street const & StreetVicinityLoader::GetStreet(uint32_t fe
void StreetVicinityLoader::LoadStreet(uint32_t featureId, Street & street)
{
FeatureType feature;
m_context->m_vector.GetByIndex(featureId, feature);
m_context->GetFeature(featureId, feature);
if (feature.GetFeatureType() != feature::GEOM_LINE)
return;

View file

@ -37,6 +37,9 @@ public:
m2::RectD m_rect;
unique_ptr<ProjectionOnStreetCalculator> m_calculator;
/// @todo Cache GetProjection results for features here, because
/// feature::GetCenter and ProjectionOnStreetCalculator::GetProjection are not so fast.
DISALLOW_COPY(Street);
};
@ -61,7 +64,7 @@ public:
continue;
FeatureType ft;
m_context->m_vector.GetByIndex(id, ft);
m_context->GetFeature(id, ft);
if (!calculator.GetProjection(feature::GetCenter(ft, FeatureType::WORST_GEOMETRY), proj))
continue;