Handle area hotels.

This commit is contained in:
Sergey Magidovich 2016-08-03 17:30:56 +03:00
parent 177a515c85
commit 67c3ee92cb
12 changed files with 133 additions and 41 deletions

View file

@ -166,14 +166,14 @@ vector<size_t> BookingDataset::GetNearestHotels(double lat, double lon, size_t l
return indexes;
}
void BookingDataset::BuildFeature(FeatureBuilder1 const & /*fb*/, size_t const hotelIndex,
void BookingDataset::BuildFeature(size_t const hotelIndex,
function<void(FeatureBuilder1 &)> const & fn) const
{
auto const & hotel = m_hotels[hotelIndex];
FeatureBuilder1 bookingFb;
FeatureParams params;
// TODO(mgsergio): handle areas.
bookingFb.SetCenter(MercatorBounds::FromLatLon(hotel.lat, hotel.lon));
auto & metadata = params.GetMetadata();
@ -333,9 +333,6 @@ bool BookingDataset::CanBeBooking(FeatureBuilder1 const & fb) const
{
// TODO(mgsergio): Remove me after refactoring is done and tested.
// Or remove the entire filter func.
if (fb.GetGeomType() != feature::GEOM_POINT)
return false;
if (fb.GetName(StringUtf8Multilang::kDefaultCode).empty())
return false;

View file

@ -88,8 +88,7 @@ public:
double maxDistance = 0.0) const;
bool MatchByName(string const & osmName, vector<size_t> const & bookingIndexes) const;
void BuildFeature(FeatureBuilder1 const & fb, size_t hotelIndex,
function<void(FeatureBuilder1 &)> const & fn) const;
void BuildFeature(size_t hotelIndex, function<void(FeatureBuilder1 &)> const & fn) const;
protected:
vector<Hotel> m_hotels;

View file

@ -51,6 +51,13 @@ string PrintBuilder(FeatureBuilder1 const & fb)
auto const center = MercatorBounds::ToLatLon(fb.GetKeyPoint());
s << "lat: " << center.lat << " lon: " << center.lon << '\t';
if (fb.GetGeomType() == feature::GEOM_POINT)
s << "GeomType: GEOM_POINT";
else if (fb.GetGeomType() == feature::GEOM_AREA)
s << "GeomType: GEOM_AREA";
else
CHECK(false, ());
return s.str();
}

View file

@ -21,7 +21,7 @@ namespace booking_scoring
namespace
{
// Calculated with tools/python/booking_hotels_quality.py.
double constexpr kOptimalThreshold = 0.321098;
double constexpr kOptimalThreshold = 0.304875;
template <typename T, typename U>
struct decay_equiv :

View file

@ -469,6 +469,25 @@ osm::Id FeatureBuilder1::GetLastOsmId() const
return m_osmIds.back();
}
osm::Id FeatureBuilder1::GetMostGenericOsmId() const
{
ASSERT(!m_osmIds.empty(), ());
auto result = m_osmIds.front();
for (auto const & id : m_osmIds)
{
if (id.IsRelation())
{
result = id;
break;
}
else if (result.IsNode() && id.IsWay())
{
result = id;
}
}
return result;
}
bool FeatureBuilder1::HasOsmId(osm::Id const & id) const
{
for (auto const & cid : m_osmIds)

View file

@ -161,12 +161,15 @@ public:
inline FeatureParams const & GetParams() const { return m_params; }
/// @name For OSM debugging, store original OSM id
/// @name For OSM debugging and osm objects replacement, store original OSM id
//@{
void AddOsmId(osm::Id id);
void SetOsmId(osm::Id id);
osm::Id GetFirstOsmId() const;
osm::Id GetLastOsmId() const;
/// @returns an id of the most general element: node's one if there is no area or relation,
/// area's one if there is no relation, and relation id otherwise.
osm::Id GetMostGenericOsmId() const;
bool HasOsmId(osm::Id const & id) const;
string GetOsmIdsString() const;
//@}

View file

@ -122,6 +122,23 @@ UNIT_TEST(FBuilder_Waterfall)
TEST_EQUAL(fb2.GetTypesCount(), 1, ());
}
UNIT_TEST(FBbuilder_GetMostGeneralOsmId)
{
FeatureBuilder1 fb;
fb.AddOsmId(osm::Id::Node(1));
TEST_EQUAL(fb.GetMostGenericOsmId(), osm::Id::Node(1), ());
fb.AddOsmId(osm::Id::Node(2));
fb.AddOsmId(osm::Id::Way(1));
TEST_EQUAL(fb.GetMostGenericOsmId(), osm::Id::Way(1), ());
fb.AddOsmId(osm::Id::Node(3));
fb.AddOsmId(osm::Id::Way(2));
fb.AddOsmId(osm::Id::Relation(1));
TEST_EQUAL(fb.GetMostGenericOsmId(), osm::Id::Relation(1), ());
}
UNIT_TEST(FVisibility_RemoveNoDrawableTypes)
{
classificator::Load();

View file

@ -38,11 +38,21 @@ uint64_t Id::OsmId() const
return m_encodedId & RESET;
}
bool Id::IsNode() const
{
return ((m_encodedId & NODE) == NODE);
}
bool Id::IsWay() const
{
return ((m_encodedId & WAY) == WAY);
}
bool Id::IsRelation() const
{
return ((m_encodedId & RELATION) == RELATION);
}
string Id::Type() const
{
if ((m_encodedId & RELATION) == RELATION)

View file

@ -21,7 +21,9 @@ public:
static Id Relation(uint64_t osmId);
uint64_t OsmId() const;
bool IsNode() const;
bool IsWay() const;
bool IsRelation() const;
/// For debug output
string Type() const;

View file

@ -185,7 +185,7 @@ public:
}
};
// TODO(mgsergio): comment
/// Used to make a "good" node for a highway graph with OSRM for low zooms.
class Place
{
FeatureBuilder1 m_ft;
@ -267,12 +267,16 @@ class MainFeaturesEmitter : public EmitterBase
unique_ptr<CoastlineFeaturesGenerator> m_coasts;
unique_ptr<feature::FeaturesCollector> m_coastsHolder;
string const m_skippedElementsPath;
ostringstream m_skippedElements;
string m_srcCoastsFile;
bool m_failOnCoasts;
generator::BookingDataset m_bookingDataset;
// TODO(mgsergio): comment.
/// Used to prepare a list of cities to serve as a list of nodes
/// for building a highway graph with OSRM for low zooms.
m4::Tree<Place> m_places;
enum TypeIndex
@ -290,7 +294,8 @@ class MainFeaturesEmitter : public EmitterBase
public:
MainFeaturesEmitter(feature::GenerateInfo const & info)
: m_failOnCoasts(info.m_failOnCoasts)
: m_skippedElementsPath(info.GetIntermediateFileName("skipped_elements", ".lst"))
, m_failOnCoasts(info.m_failOnCoasts)
, m_bookingDataset(info.m_bookingDatafileName, info.m_bookingReferenceDir)
{
Classificator const & c = classif();
@ -345,7 +350,31 @@ public:
else if ((hotelIndex = m_bookingDataset.GetMatchingHotelIndex(fb)) !=
numeric_limits<size_t>::max())
{
m_bookingDataset.BuildFeature(fb, hotelIndex, [this](FeatureBuilder1 & fb) { Emit(fb); });
m_skippedElements << DebugPrint(fb.GetMostGenericOsmId()) << endl;
// Make a hotel a simple building.
if (fb.GetGeomType() == feature::GEOM_AREA)
{
// Remove all information about a hotel.
auto params = fb.GetParams();
params.ClearName();
auto & meta = params.GetMetadata();
meta.Drop(feature::Metadata::EType::FMD_STARS);
meta.Drop(feature::Metadata::EType::FMD_WEBSITE);
meta.Drop(feature::Metadata::EType::FMD_PHONE_NUMBER);
auto & types = params.m_Types;
types.erase(remove_if(begin(types), end(types), [](uint32_t type)
{
static auto const & c = classif();
static auto tourism = c.GetTypeByPath({"tourism"});
ftype::TruncValue(type, 1);
return type == tourism;
}));
fb.SetParams(params);
Emit(fb);
}
}
else
{
@ -356,6 +385,12 @@ public:
/// @return false if coasts are not merged and FLAG_fail_on_coasts is set
bool Finish() override
{
DumpSkippedElements();
// Emit all booking objecs to the map.
for (size_t hotelIndex = 0; hotelIndex < m_bookingDataset.Size(); ++hotelIndex)
m_bookingDataset.BuildFeature(hotelIndex, [this](FeatureBuilder1 & fb) { Emit(fb); });
m_places.ForEach([this](Place const & p)
{
// m_places are no longer used after this point.
@ -459,11 +494,34 @@ private:
if (m_countries)
(*m_countries)(fb);
}
void DumpSkippedElements()
{
auto const skippedElements = m_skippedElements.str();
if (skippedElements.empty())
{
LOG(LINFO, ("No osm object was skipped."));
return;
}
ofstream file(m_skippedElementsPath, ios_base::app);
if (file.is_open())
{
file << m_skippedElements.str();
LOG(LINFO, ("Saving skipped elements to", m_skippedElementsPath, "done."));
}
else
{
LOG(LERROR, ("Can't output into", m_skippedElementsPath));
}
}
};
} // anonymous namespace
unique_ptr<EmitterBase> MakeMainFeatureEmitter(feature::GenerateInfo const & info)
{
LOG(LINFO, ("Processing booking data from", info.m_bookingDatafileName, "done."));
return make_unique<MainFeaturesEmitter>(info);
}
@ -632,22 +690,13 @@ bool GenerateFeaturesImpl(feature::GenerateInfo & info, EmitterBase & emitter)
TagReplacer tagReplacer(GetPlatform().ResourcesDir() + REPLACED_TAGS_FILE);
OsmTagMixer osmTagMixer(GetPlatform().ResourcesDir() + MIXED_TAGS_FILE);
// TODO(mgsergio): Output skipped elemnts.
// stringstream skippedElements;
// Here we can add new tags to element!!!
// Here we can add new tags to the elements!
auto const fn = [&](OsmElement * e)
{
tagReplacer(e);
tagAdmixer(e);
osmTagMixer(e);
// if (bookingDataset.BookingFilter(*e))
// {
// skippedElements << e->id << endl;
// return;
// }
parser.EmitElement(e);
};
@ -664,24 +713,6 @@ bool GenerateFeaturesImpl(feature::GenerateInfo & info, EmitterBase & emitter)
LOG(LINFO, ("Processing", info.m_osmFileName, "done."));
// TOFO(mgsergio): Build features in Emitter.
// if (!info.m_bookingDatafileName.empty())
// {
// bookingDataset.BuildFeatures([&](OsmElement * e) { parser.EmitElement(e); });
// LOG(LINFO, ("Processing booking data from", info.m_bookingDatafileName, "done."));
// string skippedElementsPath = info.GetIntermediateFileName("skipped_elements", ".lst");
// ofstream file(skippedElementsPath);
// if (file.is_open())
// {
// file << skippedElements.str();
// LOG(LINFO, ("Saving skipped elements to", skippedElementsPath, "done."));
// }
// else
// {
// LOG(LERROR, ("Can't output into", skippedElementsPath));
// }
// }
// Stop if coasts are not merged and FLAG_fail_on_coasts is set
if (!emitter.Finish())
return false;

View file

@ -239,6 +239,11 @@ bool IsDummyName(string const & s)
// FeatureParams implementation
/////////////////////////////////////////////////////////////////////////////////////////
void FeatureParams::ClearName()
{
name.Clear();
}
bool FeatureParams::AddName(string const & lang, string const & s)
{
if (IsDummyName(s))

View file

@ -222,6 +222,8 @@ public:
FeatureParams() : m_geomType(0xFF), m_reverseGeometry(false) {}
void ClearName();
bool AddName(string const & lang, string const & s);
bool AddHouseName(string const & s);
bool AddHouseNumber(string houseNumber);