Optimization step 'generate_features': Collectors can merge, new metalines builder.

This commit is contained in:
Maksim Andrianov 2019-06-28 11:40:12 +03:00 committed by mpimenov
parent 81ed74a6bb
commit 2c34655148
25 changed files with 1120 additions and 348 deletions

View file

@ -1,21 +1,52 @@
#include "generator/collector_addresses.hpp"
#include "generator/feature_builder.hpp"
#include "generator/intermediate_data.hpp"
#include "indexer/ftypes_matcher.hpp"
using namespace feature;
#include <fstream>
namespace generator
{
CollectorAddresses::CollectorAddresses(std::string const & filename)
: m_addrWriter(std::make_unique<FileWriter>(filename)) {}
: CollectorInterface(filename) {}
void CollectorAddresses::CollectFeature(FeatureBuilder const & feature, OsmElement const &)
std::shared_ptr<CollectorInterface>
CollectorAddresses::Clone(std::shared_ptr<cache::IntermediateDataReader> const &) const
{
return std::make_shared<CollectorAddresses>(GetFilename());
}
void CollectorAddresses::CollectFeature(feature::FeatureBuilder const & feature, OsmElement const &)
{
std::string addr;
auto const & checker = ftypes::IsBuildingChecker::Instance();
if (checker(feature.GetTypes()) && feature.FormatFullAddress(addr))
m_addrWriter->Write(addr.c_str(), addr.size());
m_stringStream << addr << "\n";
}
void CollectorAddresses::Save()
{
std::ofstream stream;
stream.exceptions(std::fstream::failbit | std::fstream::badbit);
stream.open(GetFilename());
stream << m_stringStream.str();
}
void CollectorAddresses::Merge(CollectorInterface const * collector)
{
CHECK(collector, ());
collector->MergeInto(const_cast<CollectorAddresses *>(this));
}
void CollectorAddresses::MergeInto(CollectorAddresses * collector) const
{
CHECK(collector, ());
collector->m_stringStream << m_stringStream.str();
}
} // namespace generator

View file

@ -2,13 +2,16 @@
#include "generator/collector_interface.hpp"
#include "coding/file_writer.hpp"
#include <memory>
#include <sstream>
#include <string>
namespace generator
{
namespace cache
{
class IntermediateDataReader;
}
// The class CollectorAddresses is responsible for the collecting addresses to the file.
class CollectorAddresses : public CollectorInterface
{
@ -16,10 +19,16 @@ public:
CollectorAddresses(std::string const & filename);
// CollectorInterface overrides:
std::shared_ptr<CollectorInterface>
Clone(std::shared_ptr<cache::IntermediateDataReader> const & = {}) const override;
void CollectFeature(feature::FeatureBuilder const & feature, OsmElement const &) override;
void Save() override {}
void Save() override;
void Merge(CollectorInterface const * collector) override;
void MergeInto(CollectorAddresses * collector) const override;
private:
std::unique_ptr<FileWriter> m_addrWriter;
std::stringstream m_stringStream;
};
} // namespace generator

View file

@ -1,6 +1,7 @@
#include "generator/collector_camera.hpp"
#include "generator/feature_builder.hpp"
#include "generator/intermediate_data.hpp"
#include "generator/osm_element.hpp"
#include "generator/maxspeeds_parser.hpp"
@ -21,8 +22,12 @@
#include "base/logging.hpp"
#include "base/string_utils.hpp"
#include <algorithm>
#include <iterator>
using namespace feature;
namespace routing
{
size_t const CameraProcessor::kMaxSpeedSpeedStringLength = 32;
@ -58,13 +63,21 @@ void CameraProcessor::ForEachCamera(Fn && toDo) const
void CameraProcessor::ProcessWay(OsmElement const & element)
{
for (auto const node : element.m_nodes)
{
if (m_speedCameras.find(node) == m_speedCameras.cend())
continue;
m_ways[element.m_id] = element.m_nodes;
}
auto & ways = m_cameraToWays[node];
ways.push_back(element.m_id);
void CameraProcessor::FillCameraInWays()
{
for (auto const & way : m_ways)
{
for (auto const & node : way.second)
{
auto const itCamera = m_speedCameras.find(node);
if (itCamera == m_speedCameras.cend())
continue;
m_cameraToWays[itCamera->first].push_back(way.first);
}
}
}
@ -75,8 +88,23 @@ void CameraProcessor::ProcessNode(OsmElement const & element)
m_speedCameras.emplace(element.m_id, std::move(camera));
}
CameraCollector::CameraCollector(std::string const & writerFile) :
m_fileWriter(writerFile) {}
void CameraProcessor::Merge(CameraProcessor const & cameraProcessor)
{
auto const & otherCameras = cameraProcessor.m_speedCameras;
m_speedCameras.insert(std::begin(otherCameras), std::end(otherCameras));
auto const & otherWays = cameraProcessor.m_ways;
m_ways.insert(std::begin(otherWays), std::end(otherWays));
}
CameraCollector::CameraCollector(std::string const & filename) :
generator::CollectorInterface(filename) {}
std::shared_ptr<generator::CollectorInterface>
CameraCollector::Clone(std::shared_ptr<generator::cache::IntermediateDataReader> const &) const
{
return std::make_shared<CameraCollector>(GetFilename());
}
void CameraCollector::CollectFeature(FeatureBuilder const & feature, OsmElement const & element)
{
@ -99,7 +127,8 @@ void CameraCollector::CollectFeature(FeatureBuilder const & feature, OsmElement
}
}
void CameraCollector::Write(CameraProcessor::CameraInfo const & camera, std::vector<uint64_t> const & ways)
void CameraCollector::Write(FileWriter & writer, CameraProcessor::CameraInfo const & camera,
std::vector<uint64_t> const & ways)
{
std::string maxSpeedStringKmPH = camera.m_speed;
int32_t maxSpeedKmPH = 0;
@ -110,23 +139,40 @@ void CameraCollector::Write(CameraProcessor::CameraInfo const & camera, std::vec
uint32_t const lat =
DoubleToUint32(camera.m_lat, ms::LatLon::kMinLat, ms::LatLon::kMaxLat, kPointCoordBits);
WriteToSink(m_fileWriter, lat);
WriteToSink(writer, lat);
uint32_t const lon =
DoubleToUint32(camera.m_lon, ms::LatLon::kMinLon, ms::LatLon::kMaxLon, kPointCoordBits);
WriteToSink(m_fileWriter, lon);
WriteToSink(writer, lon);
WriteToSink(m_fileWriter, static_cast<uint32_t>(maxSpeedKmPH));
WriteToSink(writer, static_cast<uint32_t>(maxSpeedKmPH));
auto const size = static_cast<uint32_t>(ways.size());
WriteToSink(m_fileWriter, size);
WriteToSink(writer, size);
for (auto wayId : ways)
WriteToSink(m_fileWriter, wayId);
WriteToSink(writer, wayId);
}
void CameraCollector::Save()
{
LOG(LINFO, ("void CameraCollector::Save()"));
using namespace std::placeholders;
m_processor.ForEachCamera(std::bind(&CameraCollector::Write, this, _1, _2));
m_processor.FillCameraInWays();
FileWriter writer(GetFilename());
m_processor.ForEachCamera(std::bind(&CameraCollector::Write, this, std::ref(writer), _1, _2));
}
void CameraCollector::Merge(generator::CollectorInterface const * collector)
{
CHECK(collector, ());
collector->MergeInto(const_cast<CameraCollector *>(this));
}
void CameraCollector::MergeInto(CameraCollector * collector) const
{
CHECK(collector, ());
collector->m_processor.Merge(this->m_processor);
}
} // namespace routing

View file

@ -15,6 +15,13 @@ namespace generator_tests
{
class TestCameraCollector;
} // namespace generator_tests
namespace generator
{
namespace cache
{
class IntermediateDataReader;
} // namespace cache
} // namespace generator
struct OsmElement;
@ -58,7 +65,12 @@ public:
void ProcessNode(OsmElement const & element);
void ProcessWay(OsmElement const & element);
void FillCameraInWays();
void Merge(CameraProcessor const & cameraProcessor);
private:
std::unordered_map<uint64_t, std::vector<uint64_t>> m_ways;
std::unordered_map<uint64_t, CameraInfo> m_speedCameras;
std::unordered_map<uint64_t, std::vector<uint64_t>> m_cameraToWays;
};
@ -68,18 +80,24 @@ class CameraCollector : public generator::CollectorInterface
public:
friend class generator_tests::TestCameraCollector;
explicit CameraCollector(std::string const & writerFile);
explicit CameraCollector(std::string const & filename);
// generator::CollectorInterface overrides:
std::shared_ptr<CollectorInterface>
Clone(std::shared_ptr<generator::cache::IntermediateDataReader> const & = {}) const override;
// We will process all nodes before ways because of o5m format:
// all nodes are first, then all ways, then all relations.
void CollectFeature(feature::FeatureBuilder const & feature, OsmElement const & element) override;
void Save() override;
private:
void Write(CameraProcessor::CameraInfo const & camera, std::vector<uint64_t> const & ways);
void Merge(generator::CollectorInterface const * collector) override;
void MergeInto(CameraCollector * collector) const override;
FileWriter m_fileWriter;
private:
void Write(FileWriter & writer, CameraProcessor::CameraInfo const & camera,
std::vector<uint64_t> const & ways);
std::vector<uint8_t> m_buffer;
CameraProcessor m_processor;
};
} // namespace routing

View file

@ -0,0 +1,54 @@
#include "generator/collector_city_boundary.hpp"
#include "generator/feature_generator.hpp"
#include "generator/intermediate_data.hpp"
#include "indexer/ftypes_matcher.hpp"
#include <algorithm>
#include <iterator>
using namespace feature;
namespace generator
{
CityBoundaryCollector::CityBoundaryCollector(std::string const & filename)
: CollectorInterface(filename) {}
std::shared_ptr<CollectorInterface>
CityBoundaryCollector::Clone(std::shared_ptr<cache::IntermediateDataReader> const &) const
{
return std::make_shared<CityBoundaryCollector>(GetFilename());
}
void CityBoundaryCollector::CollectFeature(FeatureBuilder const & feature, OsmElement const &)
{
if (feature.IsArea() && ftypes::IsCityTownOrVillage(feature.GetTypes()))
m_boundaries.emplace_back(feature);
}
void CityBoundaryCollector::Save()
{
FeatureBuilderWriter<serialization_policy::MaxAccuracy> collector(GetFilename());
for (auto & boundary : m_boundaries)
{
if (boundary.PreSerialize())
collector.Write(boundary);
}
}
void CityBoundaryCollector::Merge(generator::CollectorInterface const * collector)
{
CHECK(collector, ());
collector->MergeInto(const_cast<CityBoundaryCollector *>(this));
}
void CityBoundaryCollector::MergeInto(CityBoundaryCollector * collector) const
{
CHECK(collector, ());
std::copy(std::begin(m_boundaries), std::end(m_boundaries),
std::back_inserter(collector->m_boundaries));
}
} // namespace generator

View file

@ -0,0 +1,34 @@
#pragma once
#include "generator/collector_interface.hpp"
#include "generator/feature_builder.hpp"
#include <memory>
#include <vector>
namespace generator
{
namespace cache
{
class IntermediateDataReader;
} // namespace cache
class CityBoundaryCollector : public CollectorInterface
{
public:
explicit CityBoundaryCollector(std::string const & filename);
// CollectorInterface overrides:
std::shared_ptr<CollectorInterface>
Clone(std::shared_ptr<cache::IntermediateDataReader> const & = {}) const override;
void CollectFeature(feature::FeatureBuilder const & feature, OsmElement const &) override;
void Save() override;
void Merge(generator::CollectorInterface const * collector) override;
void MergeInto(CityBoundaryCollector * collector) const override;
private:
std::vector<feature::FeatureBuilder> m_boundaries;
};
} // namespace generator

View file

@ -8,6 +8,15 @@ using namespace feature;
namespace generator
{
std::shared_ptr<CollectorInterface>
CollectorCollection::Clone(std::shared_ptr<cache::IntermediateDataReader> const & cache) const
{
auto p = std::make_shared<CollectorCollection>();
for (auto const & c : m_collection)
p->Append(c->Clone(cache));
return p;
}
void CollectorCollection::Collect(OsmElement const & element)
{
for (auto & c : m_collection)
@ -31,4 +40,21 @@ void CollectorCollection::Save()
for (auto & c : m_collection)
c->Save();
}
void CollectorCollection::Merge(CollectorInterface const * collector)
{
CHECK(collector, ());
collector->MergeInto(const_cast<CollectorCollection *>(this));
}
void CollectorCollection::MergeInto(CollectorCollection * collector) const
{
CHECK(collector, ());
auto & otherCollection = collector->m_collection;
CHECK_EQUAL(m_collection.size(), otherCollection.size(), ());
for (size_t i = 0; i < m_collection.size(); ++i)
otherCollection[i]->Merge(m_collection[i].get());
}
} // namespace generator

View file

@ -15,14 +15,25 @@ class FeatureBuilder;
namespace generator
{
namespace cache
{
class IntermediateDataReader;
} // namespace cache
// This class allows you to work with a group of collectors as with one.
class CollectorCollection : public CollectionBase<std::shared_ptr<CollectorInterface>>, public CollectorInterface
{
public:
// CollectorInterface overrides:
std::shared_ptr<CollectorInterface>
Clone(std::shared_ptr<cache::IntermediateDataReader> const & cache = {}) const override;
void Collect(OsmElement const & element) override;
void CollectRelation(RelationElement const & element) override;
void CollectFeature(feature::FeatureBuilder const & feature, OsmElement const & element) override;
void Save() override;
void Merge(CollectorInterface const * collector) override;
void MergeInto(CollectorCollection * collector) const override;
};
} // namespace generator

View file

@ -1,5 +1,8 @@
#pragma once
#include "base/assert.hpp"
#include <memory>
#include <string>
struct OsmElement;
@ -14,19 +17,66 @@ namespace base
{
class GeoObjectId;
} // namespace base
namespace feature
{
class MetalinesBuilder;
} // namespace feature
namespace routing
{
class CameraCollector;
class RestrictionWriter;
class RoadAccessWriter;
} // namespace routing
namespace generator
{
class CollectorAddresses;
class CollectorCollection;
class CollectorTag;
class MaxspeedsCollector;
class CityBoundaryCollector;
namespace cache
{
class IntermediateDataReader;
} // namespace cache
namespace regions
{
class CollectorRegionInfo;
} // namespace regions
// Implementing this interface allows an object to collect data from RelationElement,
// OsmElement and FeatureBuilder1 elements.
// OsmElement and FeatureBuilder elements.
class CollectorInterface
{
public:
CollectorInterface(std::string const & filename = {}) : m_filename(filename) {}
virtual ~CollectorInterface() = default;
virtual std::shared_ptr<CollectorInterface>
Clone(std::shared_ptr<cache::IntermediateDataReader> const & = {}) const = 0;
virtual void Collect(OsmElement const &) {}
virtual void CollectRelation(RelationElement const &) {}
virtual void CollectFeature(feature::FeatureBuilder const &, OsmElement const &) {}
virtual void Save() = 0;
virtual void Merge(CollectorInterface const *) = 0;
virtual void MergeInto(CityBoundaryCollector *) const { FailIfMethodUnsuppirted(); }
virtual void MergeInto(routing::CameraCollector *) const { FailIfMethodUnsuppirted(); }
virtual void MergeInto(routing::RestrictionWriter *) const { FailIfMethodUnsuppirted(); }
virtual void MergeInto(routing::RoadAccessWriter *) const { FailIfMethodUnsuppirted(); }
virtual void MergeInto(CollectorAddresses *) const { FailIfMethodUnsuppirted(); }
virtual void MergeInto(CollectorTag *) const { FailIfMethodUnsuppirted(); }
virtual void MergeInto(MaxspeedsCollector *) const { FailIfMethodUnsuppirted(); }
virtual void MergeInto(feature::MetalinesBuilder *) const { FailIfMethodUnsuppirted(); }
virtual void MergeInto(regions::CollectorRegionInfo *) const { FailIfMethodUnsuppirted(); }
virtual void MergeInto(CollectorCollection *) const { FailIfMethodUnsuppirted(); }
std::string const & GetFilename() const { return m_filename; }
private:
void FailIfMethodUnsuppirted() const { CHECK(false, ("This method is unsupported.")); }
std::string m_filename;
};
} // namespace generator

View file

@ -1,5 +1,6 @@
#include "generator/collector_tag.hpp"
#include "generator/intermediate_data.hpp"
#include "generator/osm_element.hpp"
#include "base/geo_object_id.hpp"
@ -9,19 +10,38 @@ namespace generator
{
CollectorTag::CollectorTag(std::string const & filename, std::string const & tagKey,
Validator const & validator, bool ignoreIfNotOpen)
: m_tagKey(tagKey), m_validator(validator), m_needCollect(true)
: CollectorInterface(filename)
, m_tagKey(tagKey)
, m_validator(validator)
, m_ignoreIfNotOpen(ignoreIfNotOpen) {}
std::shared_ptr<CollectorInterface>
CollectorTag::Clone(std::shared_ptr<cache::IntermediateDataReader> const &) const
{
m_stream.exceptions(std::fstream::failbit | std::fstream::badbit);
return std::make_shared<CollectorTag>(GetFilename(), m_tagKey, m_validator, m_ignoreIfNotOpen);
}
void CollectorTag::Collect(OsmElement const & el)
{
auto const tag = el.GetTag(m_tagKey);
if (!tag.empty() && m_validator(tag))
m_stream << GetGeoObjectId(el).GetEncodedId() << "\t" << tag << "\n";
}
void CollectorTag::Save()
{
std::ofstream stream;
stream.exceptions(std::fstream::failbit | std::fstream::badbit);
try
{
m_stream.open(filename);
stream.open(GetFilename());
stream << m_stream.str();
}
catch (std::ios::failure const & e)
{
if (ignoreIfNotOpen)
if (m_ignoreIfNotOpen)
{
m_needCollect = false;
LOG(LINFO, ("Сould not open file", filename, ". This was ignored."));
LOG(LINFO, ("Сould not open file", GetFilename(), ". This was ignored."));
}
else
{
@ -30,13 +50,17 @@ CollectorTag::CollectorTag(std::string const & filename, std::string const & tag
}
}
void CollectorTag::Collect(OsmElement const & el)
void CollectorTag::Merge(CollectorInterface const * collector)
{
if (!m_needCollect)
return;
CHECK(collector, ());
auto const tag = el.GetTag(m_tagKey);
if (!tag.empty() && m_validator(tag))
m_stream << GetGeoObjectId(el).GetEncodedId() << "\t" << tag << "\n";
collector->MergeInto(const_cast<CollectorTag *>(this));
}
void CollectorTag::MergeInto(CollectorTag * collector) const
{
CHECK(collector, ());
collector->m_stream << this->m_stream.str();
}
} // namespace generator

View file

@ -4,6 +4,8 @@
#include <fstream>
#include <functional>
#include <memory>
#include <sstream>
#include <string>
struct OsmElement;
@ -14,6 +16,11 @@ class GeoObjectId;
namespace generator
{
namespace cache
{
class IntermediateDataReader;
} // namespace cache
// CollectorTag class collects validated value of a tag and saves it to file with following
// format: osmId<tab>tagValue.
class CollectorTag : public CollectorInterface
@ -25,13 +32,18 @@ public:
Validator const & validator, bool ignoreIfNotOpen = false);
// CollectorInterface overrides:
void Collect(OsmElement const & el) override;
void Save() override {}
std::shared_ptr<CollectorInterface>
Clone(std::shared_ptr<cache::IntermediateDataReader> const & = {}) const override;
void Collect(OsmElement const & el) override;
void Save() override;
void Merge(CollectorInterface const * collector) override;
void MergeInto(CollectorTag * collector) const override;
private:
std::ofstream m_stream;
std::stringstream m_stream;
std::string m_tagKey;
Validator m_validator;
bool m_needCollect;
bool m_ignoreIfNotOpen;
};
} // namespace generator

View file

@ -135,6 +135,7 @@ public:
}
bool HasType(uint32_t t) const { return m_params.IsTypeExist(t); }
bool HasType(uint32_t t, uint8_t level) const { return m_params.IsTypeExist(t, level); }
uint32_t FindType(uint32_t comp, uint8_t level) const { return m_params.FindType(comp, level); }
FeatureParams::Types const & GetTypes() const { return m_params.m_types; }

View file

@ -17,7 +17,9 @@ set(
filter_elements_tests.cpp
intermediate_data_test.cpp
maxspeeds_tests.cpp
merge_collectors_tests.cpp
metadata_parser_test.cpp
metalines_tests.cpp
node_mixer_test.cpp
osm2meta_test.cpp
osm_o5m_source_test.cpp

View file

@ -0,0 +1,92 @@
#include "testing/testing.hpp"
#include "generator/generator_tests/common.hpp"
#include "generator/collector_collection.hpp"
#include "generator/collector_interface.hpp"
#include "generator/collector_tag.hpp"
#include "base/geo_object_id.hpp"
#include "platform/platform.hpp"
#include <fstream>
#include <memory>
#include <string>
#include <vector>
using namespace base;
using namespace generator;
using namespace generator_tests;
namespace
{
auto const kEmptyValidator = [](auto const &) { return true; };
}
UNIT_TEST(MergeCollector_Case1)
{
auto const filename = GetFileName();
std::string const tagKey = "admin_level";
auto collector1 = std::make_shared<CollectorTag>(filename, tagKey, kEmptyValidator);
collector1->Collect(MakeOsmElement(1 /* id */, {{"admin_level", "1"}}, OsmElement::EntityType::Relation));
collector1->Collect(MakeOsmElement(2 /* id */, {{"admin_level", "2"}}, OsmElement::EntityType::Relation));
auto collector2 = collector1->Clone();
collector2->Collect(MakeOsmElement(3 /* id */, {{"admin_level", "3"}}, OsmElement::EntityType::Relation));
collector2->Collect(MakeOsmElement(4 /* id */, {{"admin_level", "4"}}, OsmElement::EntityType::Relation));
collector1->Merge(collector2.get());
collector1->Save();
std::vector<std::string> const answers = {
std::to_string(GeoObjectId(GeoObjectId::Type::ObsoleteOsmRelation, 1 /* id */).GetEncodedId()) + "\t1",
std::to_string(GeoObjectId(GeoObjectId::Type::ObsoleteOsmRelation, 2 /* id */).GetEncodedId()) + "\t2",
std::to_string(GeoObjectId(GeoObjectId::Type::ObsoleteOsmRelation, 3 /* id */).GetEncodedId()) + "\t3",
std::to_string(GeoObjectId(GeoObjectId::Type::ObsoleteOsmRelation, 4 /* id */).GetEncodedId()) + "\t4",
};
std::ifstream stream;
stream.exceptions(std::ios::badbit);
stream.open(filename);
size_t pos = 0;
std::string line;
while (std::getline(stream, line))
TEST_EQUAL(line, answers[pos++], ());
}
UNIT_TEST(MergeCollector_Case2)
{
auto const filename = GetFileName();
std::string const tagKey = "admin_level";
auto collection1 = std::make_shared<CollectorCollection>();
collection1->Append(std::make_shared<CollectorTag>(filename, tagKey, kEmptyValidator));
collection1->Collect(MakeOsmElement(1 /* id */, {{"admin_level", "1"}}, OsmElement::EntityType::Relation));
collection1->Collect(MakeOsmElement(2 /* id */, {{"admin_level", "2"}}, OsmElement::EntityType::Relation));
auto collection2 = collection1->Clone();
collection2->Collect(MakeOsmElement(3 /* id */, {{"admin_level", "3"}}, OsmElement::EntityType::Relation));
collection2->Collect(MakeOsmElement(4 /* id */, {{"admin_level", "4"}}, OsmElement::EntityType::Relation));
collection1->Merge(collection2.get());
collection1->Save();
std::vector<std::string> const answers = {
std::to_string(GeoObjectId(GeoObjectId::Type::ObsoleteOsmRelation, 1 /* id */).GetEncodedId()) + "\t1",
std::to_string(GeoObjectId(GeoObjectId::Type::ObsoleteOsmRelation, 2 /* id */).GetEncodedId()) + "\t2",
std::to_string(GeoObjectId(GeoObjectId::Type::ObsoleteOsmRelation, 3 /* id */).GetEncodedId()) + "\t3",
std::to_string(GeoObjectId(GeoObjectId::Type::ObsoleteOsmRelation, 4 /* id */).GetEncodedId()) + "\t4",
};
std::ifstream stream;
stream.exceptions(std::ios::badbit);
stream.open(filename);
size_t pos = 0;
std::string line;
while (std::getline(stream, line))
TEST_EQUAL(line, answers[pos++], ());
}

View file

@ -0,0 +1,148 @@
#include "testing/testing.hpp"
#include "generator/metalines_builder.hpp"
#include "generator/osm_element.hpp"
#include <cstddef>
#include <cstdint>
#include <functional>
#include <memory>
#include <string>
#include <vector>
using namespace feature;
namespace
{
OsmElement MakeHighway(uint64_t id, std::string const & name, std::vector<uint64_t> const & nodes, bool isOneway = false)
{
OsmElement element;
element.m_id = id;
element.m_type = OsmElement::EntityType::Way;
element.AddTag("ref", "");
element.AddTag("name", name);
element.AddTag("highway", "primary");
if (isOneway)
element.AddTag("oneway", "yes");
element.m_nodes = nodes;
return element;
}
size_t MakeKey(OsmElement const & element)
{
auto const name = element.GetTag("name");
auto const ref = element.GetTag("ref");
return std::hash<std::string>{}(name + '\0' + ref);
}
LineStringMerger::InputData MakeInputData(std::vector<OsmElement> const & elements)
{
LineStringMerger::InputData inputData;
for (auto const & element : elements)
inputData.emplace(MakeKey(element), std::make_shared<LineString>(element));
return inputData;
};
bool IsEquil(LineStringMerger::LinePtr const & lineString, std::vector<int32_t> const & ways)
{
auto const & w = lineString->GetWays();
return w == ways;
}
auto const w1 = MakeHighway(1/* id */, "w" /* name */, {1, 2, 3} /* nodes */);
auto const w2 = MakeHighway(2/* id */, "w" /* name */, {3, 4, 5} /* nodes */);
auto const w3 = MakeHighway(3/* id */, "w" /* name */, {5, 6, 7} /* nodes */);
auto const w4 = MakeHighway(4/* id */, "w" /* name */, {7, 8, 9} /* nodes */);
auto const w5 = MakeHighway(5/* id */, "w" /* name */, {9, 10, 11} /* nodes */);
auto const wo6 = MakeHighway(6/* id */, "w" /* name */, {13, 12, 3} /* nodes */, true /* isOneway */);
auto const wo7 = MakeHighway(7/* id */, "w" /* name */, {15, 14, 13} /* nodes */, true /* isOneway */);
auto const wo8 = MakeHighway(8/* id */, "w" /* name */, {17, 16, 15} /* nodes */, true /* isOneway */);
auto const b1 = MakeHighway(1/* id */, "b" /* name */, {1, 2, 3} /* nodes */);
auto const b2 = MakeHighway(2/* id */, "b" /* name */, {3, 4, 5} /* nodes */);
} // namespace
UNIT_TEST(MetalinesTest_Case0)
{
auto const inputData = MakeInputData({w1});
auto outputData = LineStringMerger::Merge(inputData);
TEST_EQUAL(outputData.size(), 0 /* unique names roads count */, ());
outputData = LineStringMerger::Merge({});
TEST_EQUAL(outputData.size(), 0 /* unique names roads count */, ());
}
UNIT_TEST(MetalinesTest_Case1)
{
auto const inputData = MakeInputData({w1, w2});
auto const outputData = LineStringMerger::Merge(inputData);
auto const key = MakeKey(w1);
TEST_EQUAL(outputData.size(), 1 /* unique names roads count */, ());
TEST_EQUAL(outputData.at(key)[0]->GetWays().size(), 2 /* merged way size */, ());
TEST(IsEquil(outputData.at(key)[0], {1, 2}) /* merged way */, ());
}
UNIT_TEST(MetalinesTest_Case2)
{
auto const inputData = MakeInputData({w1, w3, w2});
auto const outputData = LineStringMerger::Merge(inputData);
auto const key = MakeKey(w1);
TEST_EQUAL(outputData.size(), 1 /* unique names roads count */, ());
TEST_EQUAL(outputData.at(key)[0]->GetWays().size(), 3 /* merged way size */, ());
TEST(IsEquil(outputData.at(key)[0], {1, 2, 3}) /* merged way */, ());
}
UNIT_TEST(MetalinesTest_Case3)
{
auto const inputData = MakeInputData({w1, w4, w2, w5,});
auto const outputData = LineStringMerger::Merge(inputData);
auto const key = MakeKey(w1);
TEST_EQUAL(outputData.size(), 1 /* unique names roads count */, ());
TEST_EQUAL(outputData.at(key).size(), 2 /* ways count */, ());
TEST_EQUAL(outputData.at(key)[0]->GetWays().size(), 2 /* merged way size */, ());
TEST(IsEquil(outputData.at(key)[0], {1, 2}) /* merged way */, ());
TEST_EQUAL(outputData.at(key)[1]->GetWays().size(), 2 /* merged way size */, ());
TEST(IsEquil(outputData.at(key)[1], {4, 5}) /* merged way */, ());
}
UNIT_TEST(MetalinesTest_Case4)
{
auto const inputData = MakeInputData({w1, wo6,});
auto const outputData = LineStringMerger::Merge(inputData);
auto const key = MakeKey(w1);
TEST_EQUAL(outputData.size(), 1 /* unique names roads count */, ());
TEST_EQUAL(outputData.at(key).size(), 1 /* ways count */, ());
TEST(IsEquil(outputData.at(key)[0], {6, -1}) /* merged way */, ());
}
UNIT_TEST(MetalinesTest_Case5)
{
auto const inputData = MakeInputData({w1, w2, wo6,});
auto const outputData = LineStringMerger::Merge(inputData);
auto const key = MakeKey(w1);
TEST_EQUAL(outputData.size(), 1 /* unique names roads count */, ());
TEST_EQUAL(outputData.at(key).size(), 1 /* ways count */, ());
TEST(IsEquil(outputData.at(key)[0], {1, 2}) /* merged way */, ());
}
UNIT_TEST(MetalinesTest_Case6)
{
auto const inputData = MakeInputData({w1, b1, w2, b2,});
auto const outputData = LineStringMerger::Merge(inputData);
auto const keyW = MakeKey(w1);
auto const keyB = MakeKey(b1);
TEST_EQUAL(outputData.size(), 2 /* unique names roads count */, ());
TEST_EQUAL(outputData.at(keyW).size(), 1 /* ways count */, ());
TEST_EQUAL(outputData.at(keyB).size(), 1 /* ways count */, ());
}

View file

@ -10,7 +10,9 @@
#include "base/logging.hpp"
#include "base/string_utils.hpp"
#include <algorithm>
#include <fstream>
#include <iterator>
#include <sstream>
using namespace base;
@ -33,6 +35,16 @@ bool ParseMaxspeedAndWriteToStream(string const & maxspeed, SpeedInUnits & speed
namespace generator
{
MaxspeedsCollector::MaxspeedsCollector(string const & filename)
: CollectorInterface(filename) {}
std::shared_ptr<CollectorInterface>
MaxspeedsCollector::Clone(std::shared_ptr<cache::IntermediateDataReader> const &) const
{
return std::make_shared<MaxspeedsCollector>(GetFilename());
}
void MaxspeedsCollector::CollectFeature(FeatureBuilder const &, OsmElement const & p)
{
if (!p.IsWay())
@ -101,26 +113,29 @@ void MaxspeedsCollector::CollectFeature(FeatureBuilder const &, OsmElement const
void MaxspeedsCollector::Save()
{
Flush();
}
LOG(LINFO, ("Saving maxspeed tag values to", GetFilename()));
void MaxspeedsCollector::Flush()
{
LOG(LINFO, ("Saving maxspeed tag values to", m_filePath));
ofstream stream(m_filePath);
if (!stream)
{
LOG(LERROR, ("Cannot open file", m_filePath));
return;
}
ofstream stream;
stream.exceptions(fstream::failbit | fstream::badbit);
stream.open(GetFilename());
for (auto const & s : m_data)
stream << s << '\n';
if (stream.fail())
LOG(LERROR, ("Cannot write to file", m_filePath));
else
LOG(LINFO, ("Wrote", m_data.size(), "maxspeed tags to", m_filePath));
LOG(LINFO, ("Wrote", m_data.size(), "maxspeed tags to", GetFilename()));
}
void MaxspeedsCollector::Merge(CollectorInterface const * collector)
{
CHECK(collector, ());
collector->MergeInto(const_cast<MaxspeedsCollector *>(this));
}
void MaxspeedsCollector::MergeInto(MaxspeedsCollector * collector) const
{
CHECK(collector, ());
copy(begin(m_data), end(m_data), back_inserter(collector->m_data));
}
} // namespace generator

View file

@ -1,13 +1,20 @@
#pragma once
#include "generator/collector_interface.hpp"
#include "generator/feature_builder.hpp"
#include "generator/osm_element.hpp"
#include <memory>
#include <string>
#include <vector>
namespace generator
{
namespace cache
{
class IntermediateDataReader;
} // namespace cache
/// \brief Collects all maxspeed tags value and saves them to a csv file.
/// Every line describes maxspeed, maxspeed:forward and maxspeed:backward
/// tags of features. The format of the lines is described below.
@ -15,15 +22,19 @@ class MaxspeedsCollector : public CollectorInterface
{
public:
/// \param filePath path to csv file.
explicit MaxspeedsCollector(std::string const & filePath) : m_filePath(filePath) {}
explicit MaxspeedsCollector(std::string const & filename);
// CollectorInterface overrides:
std::shared_ptr<CollectorInterface>
Clone(std::shared_ptr<cache::IntermediateDataReader> const & = {}) const override;
void CollectFeature(feature::FeatureBuilder const &, OsmElement const & p) override;
void Save() override;
private:
void Flush();
void Merge(CollectorInterface const * collector) override;
void MergeInto(MaxspeedsCollector * collector) const override;
private:
// |m_data| contains strings with maxspeed tags value for corresponding features in one of the
// following formats
// 1. osm id,units kmh or mph,maxspeed value
@ -47,6 +58,5 @@ private:
// are converted to an appropriate speed value and macro "none" and "walk" are converted
// to |kNoneMaxSpeed| and |kWalkMaxSpeed|.
std::vector<std::string> m_data;
std::string m_filePath;
};
} // namespace generator

View file

@ -1,5 +1,6 @@
#include "generator/metalines_builder.hpp"
#include "generator/intermediate_data.hpp"
#include "generator/routing_helpers.hpp"
#include "indexer/classificator.hpp"
@ -20,181 +21,208 @@
namespace
{
uint8_t const kMetaLinesSectionVersion = 1;
using Ways = std::vector<int32_t>;
/// A string of connected ways.
class LineString
{
Ways m_ways;
uint64_t m_start;
uint64_t m_end;
bool m_oneway;
public:
explicit LineString(OsmElement const & way)
{
std::string const oneway = way.GetTag("oneway");
m_oneway = !oneway.empty() && oneway != "no";
int32_t const wayId = base::checked_cast<int32_t>(way.m_id);
m_ways.push_back(oneway == "-1" ? -wayId : wayId);
CHECK_GREATER_OR_EQUAL(way.Nodes().size(), 2, ());
m_start = way.Nodes().front();
m_end = way.Nodes().back();
}
Ways const & GetWays() const { return m_ways; }
void Reverse()
{
ASSERT(!m_oneway, ("Trying to reverse a one-way road."));
std::swap(m_start, m_end);
std::reverse(m_ways.begin(), m_ways.end());
for (auto & p : m_ways)
p = -p;
}
bool Add(LineString & line)
{
if (m_start == line.m_start || m_end == line.m_end)
{
if (!line.m_oneway)
line.Reverse();
else if (!m_oneway)
Reverse();
else
return false;
}
if (m_end == line.m_start)
{
m_ways.insert(m_ways.end(), line.m_ways.begin(), line.m_ways.end());
m_end = line.m_end;
m_oneway = m_oneway || line.m_oneway;
}
else if (m_start == line.m_end)
{
m_ways.insert(m_ways.begin(), line.m_ways.begin(), line.m_ways.end());
m_start = line.m_start;
m_oneway = m_oneway || line.m_oneway;
}
else
{
return false;
}
return true;
}
};
} // namespace
namespace feature
{
/// A list of segments, that is, LineStrings, sharing the same attributes.
class Segments
LineString::LineString(OsmElement const & way)
{
std::list<LineString> m_parts;
std::string const oneway = way.GetTag("oneway");
m_oneway = !oneway.empty() && oneway != "no";
int32_t const wayId = base::checked_cast<int32_t>(way.m_id);
m_ways.push_back(oneway == "-1" ? -wayId : wayId);
CHECK_GREATER_OR_EQUAL(way.Nodes().size(), 2, ());
m_start = way.Nodes().front();
m_end = way.Nodes().back();
}
public:
explicit Segments(OsmElement const & way) { m_parts.emplace_back(way); }
void LineString::Reverse()
{
ASSERT(!m_oneway, ("Trying to reverse a one-way road."));
std::swap(m_start, m_end);
std::reverse(m_ways.begin(), m_ways.end());
for (auto & p : m_ways)
p = -p;
}
void Add(OsmElement const & way)
bool LineString::Add(LineString & line)
{
if (m_start == line.m_start || m_end == line.m_end)
{
LineString line(way);
auto found = m_parts.end();
for (auto i = m_parts.begin(); i != m_parts.end(); ++i)
if (!line.m_oneway)
line.Reverse();
else if (!m_oneway)
Reverse();
else
return false;
}
if (m_end == line.m_start)
{
m_ways.insert(m_ways.end(), line.m_ways.begin(), line.m_ways.end());
m_end = line.m_end;
m_oneway = m_oneway || line.m_oneway;
}
else if (m_start == line.m_end)
{
m_ways.insert(m_ways.begin(), line.m_ways.begin(), line.m_ways.end());
m_start = line.m_start;
m_oneway = m_oneway || line.m_oneway;
}
else
{
return false;
}
return true;
}
// static
LineStringMerger::OutputData LineStringMerger::Merge(InputData const & data)
{
InputData mergedLines;
auto const intermediateData = OrderData(data);
for (auto & p : intermediateData)
{
Buffer buffer;
for (auto & lineString : p.second)
TryMerge(lineString, buffer);
std::unordered_set<LinePtr> uniqLineStrings;
for (auto const & pb : buffer)
{
if (i->Add(line))
{
found = i;
break;
}
}
// If no LineString accepted the way in its Add method, create a new LineString with it.
if (found == m_parts.cend())
{
m_parts.push_back(line);
return;
}
// Otherwise check if the extended LineString can be merged with some other LineString.
for (LineString & part : m_parts)
{
if (part.Add(*found))
{
m_parts.erase(found);
break;
}
auto const & ways = pb.second->GetWays();
if (uniqLineStrings.emplace(pb.second).second && ways.size() > 1)
mergedLines.emplace(p.first, pb.second);
}
}
std::vector<Ways> GetLongWays() const
return OrderData(mergedLines);
}
// static
bool LineStringMerger::TryMerge(LinePtr const & lineString, Buffer & buffer)
{
bool merged = false;
while(TryMergeOne(lineString, buffer))
merged = true;
buffer.emplace(lineString->GetStart(), lineString);
buffer.emplace(lineString->GetEnd(), lineString);
return merged;
}
// static
bool LineStringMerger::TryMergeOne(LinePtr const & lineString, Buffer & buffer)
{
auto static const kUndef = std::numeric_limits<int>::max();
uint64_t index = kUndef;
if (buffer.count(lineString->GetStart()) != 0)
index = lineString->GetStart();
else if (buffer.count(lineString->GetEnd()) != 0)
index = lineString->GetEnd();
if (index != kUndef)
{
std::vector<Ways> result;
for (LineString const & line : m_parts)
auto bufferedLineString = buffer[index];
buffer.erase(bufferedLineString->GetStart());
buffer.erase(bufferedLineString->GetEnd());
if (!lineString->Add(*bufferedLineString))
{
if (line.GetWays().size() > 1)
{
result.push_back(line.GetWays());
}
buffer.emplace(bufferedLineString->GetStart(), bufferedLineString);
buffer.emplace(bufferedLineString->GetEnd(), bufferedLineString);
buffer.emplace(lineString->GetStart(), lineString);
buffer.emplace(lineString->GetEnd(), lineString);
return false;
}
return result;
}
};
return index != kUndef;
}
// static
LineStringMerger::OutputData LineStringMerger::OrderData(InputData const & data)
{
OutputData intermediateData;
for (auto const & p : data)
intermediateData[p.first].emplace_back(p.second);
for (auto & p : intermediateData)
{
auto & lineStrings = intermediateData[p.first];
std::sort(std::begin(lineStrings), std::end(lineStrings), [](auto const & l, auto const & r) {
auto const & lways = l->GetWays();
auto const & rways = r->GetWays();
return lways.size() == rways.size() ? lways.front() < rways.front() : lways.size() > rways.size();
});
}
return intermediateData;
}
// MetalinesBuilder --------------------------------------------------------------------------------
MetalinesBuilder::MetalinesBuilder(std::string const & filename)
: generator::CollectorInterface(filename) {}
std::shared_ptr<generator::CollectorInterface>
MetalinesBuilder::Clone(std::shared_ptr<generator::cache::IntermediateDataReader> const &) const
{
return std::make_shared<MetalinesBuilder>(GetFilename());
}
void MetalinesBuilder::CollectFeature(FeatureBuilder const & feature, OsmElement const & element)
{
if (!feature.IsLine())
return;
auto const & params = feature.GetParams();
static uint32_t const highwayType = classif().GetTypeByPath({"highway"});
if (params.FindType(highwayType, 1) == ftype::GetEmptyValue() ||
element.Nodes().front() == element.Nodes().back())
{
static auto const highwayType = classif().GetTypeByPath({"highway"});
if (!feature.HasType(highwayType, 1 /* level */) || element.Nodes().front() == element.Nodes().back())
return;
}
std::string name;
params.name.GetString(StringUtf8Multilang::kDefaultCode, name);
auto const & params = feature.GetParams();
auto const name = feature.GetName();
if (name.empty() && params.ref.empty())
return;
size_t const key = std::hash<std::string>{}(name + '\0' + params.ref);
auto segment = m_data.find(key);
if (segment == m_data.cend())
m_data.emplace(key, std::make_shared<Segments>(element));
else
segment->second->Add(element);
auto const key = std::hash<std::string>{}(name + '\0' + params.ref);
m_data.emplace(key, std::make_shared<LineString>(element));
}
void MetalinesBuilder::Save()
{
Flush();
FileWriter writer(GetFilename());
uint32_t countLines = 0;
uint32_t countWays = 0;
auto const mergedData = LineStringMerger::Merge(m_data);
for (auto const & p : mergedData)
{
for (auto const & lineString : p.second)
{
auto const & ways = lineString->GetWays();
uint16_t size = base::checked_cast<uint16_t>(ways.size());
WriteToSink(writer, size);
countWays += ways.size();
for (int32_t const way : ways)
WriteToSink(writer, way);
++countLines;
}
}
LOG_SHORT(LINFO, ("Wrote", countLines, "metalines [with", countWays ,
"ways] with OSM IDs for the entire planet to", GetFilename()));
}
void MetalinesBuilder::Flush()
void MetalinesBuilder::Merge(generator::CollectorInterface const * collector)
{
try
{
uint32_t count = 0;
FileWriter writer(m_filePath);
for (auto const & seg : m_data)
{
auto const & longWays = seg.second->GetLongWays();
for (Ways const & ways : longWays)
{
uint16_t size = base::checked_cast<uint16_t>(ways.size());
WriteToSink(writer, size);
for (int32_t const way : ways)
WriteToSink(writer, way);
++count;
}
}
LOG_SHORT(LINFO, ("Wrote", count, "metalines with OSM IDs for the entire planet to", m_filePath));
}
catch (RootException const & e)
{
LOG(LERROR, ("An exception happened while saving metalines to", m_filePath, ":", e.what()));
}
CHECK(collector, ());
collector->MergeInto(const_cast<MetalinesBuilder *>(this));
}
void MetalinesBuilder::MergeInto(MetalinesBuilder * collector) const
{
CHECK(collector, ());
collector->m_data.insert(std::begin(m_data), std::end(m_data));
}
// Functions --------------------------------------------------------------------------------

View file

@ -5,35 +5,79 @@
#include "generator/osm_element.hpp"
#include <cstdlib>
#include <map>
#include <memory>
#include <string>
#include <unordered_map>
namespace generator
{
namespace cache
{
class IntermediateDataReader;
} // namespace cache
} // namespace generator
namespace feature
{
class Segments;
// A string of connected ways.
class LineString
{
public:
using Ways = std::vector<int32_t>;
/// Merges road segments with similar name and ref values into groups called metalines.
explicit LineString(OsmElement const & way);
bool Add(LineString & line);
void Reverse();
Ways const & GetWays() const { return m_ways; }
uint64_t GetStart() const { return m_start; }
uint64_t GetEnd() const { return m_end; }
private:
uint64_t m_start;
uint64_t m_end;
bool m_oneway;
Ways m_ways;
};
class LineStringMerger
{
public:
using LinePtr = std::shared_ptr<LineString>;
using InputData = std::unordered_multimap<size_t, LinePtr>;
using OutputData = std::map<size_t, std::vector<LinePtr>>;
using Buffer = std::unordered_map<uint64_t, LinePtr>;
static OutputData Merge(InputData const & data);
static bool TryMerge(LinePtr const & lineString, Buffer & buffer);
static bool TryMergeOne(LinePtr const & lineString, Buffer & buffer);
static OutputData OrderData(InputData const & data);
};
// Merges road segments with similar name and ref values into groups called metalines.
class MetalinesBuilder : public generator::CollectorInterface
{
public:
explicit MetalinesBuilder(std::string const & filePath) : m_filePath(filePath) {}
explicit MetalinesBuilder(std::string const & filename);
// CollectorInterface overrides:
std::shared_ptr<CollectorInterface>
Clone(std::shared_ptr<generator::cache::IntermediateDataReader> const & = {}) const override;
/// Add a highway segment to the collection of metalines.
void CollectFeature(FeatureBuilder const & feature, OsmElement const & element) override;
void Save() override;
/// Write all metalines to the intermediate file.
void Flush();
void Merge(generator::CollectorInterface const * collector) override;
void MergeInto(MetalinesBuilder * collector) const override;
private:
std::unordered_map<size_t, std::shared_ptr<Segments>> m_data;
std::string m_filePath;
std::unordered_multimap<size_t, std::shared_ptr<LineString>> m_data;
};
/// Read an intermediate file from MetalinesBuilder and convert it to an mwm section.
// Read an intermediate file from MetalinesBuilder and convert it to an mwm section.
bool WriteMetalinesSection(std::string const & mwmPath, std::string const & metalinesPath,
std::string const & osmIdsToFeatureIdsPath);
}

View file

@ -1,6 +1,6 @@
#include "generator/regions/collector_region_info.hpp"
#include "generator/feature_builder.hpp"
#include "generator/intermediate_data.hpp"
#include "generator/osm_element.hpp"
#include "coding/file_writer.hpp"
@ -67,7 +67,15 @@ char const * GetLabel(PlaceLevel level)
UNREACHABLE();
}
CollectorRegionInfo::CollectorRegionInfo(std::string const & filename) : m_filename(filename) {}
CollectorRegionInfo::CollectorRegionInfo(std::string const & filename)
: CollectorInterface(filename) {}
std::shared_ptr<CollectorInterface>
CollectorRegionInfo::Clone(std::shared_ptr<cache::IntermediateDataReader> const &) const
{
return std::make_shared<CollectorRegionInfo>(GetFilename());
}
void CollectorRegionInfo::Collect(OsmElement const & el)
{
@ -86,12 +94,27 @@ void CollectorRegionInfo::Collect(OsmElement const & el)
void CollectorRegionInfo::Save()
{
FileWriter writer(m_filename);
FileWriter writer(GetFilename());
WriteToSink(writer, kVersion);
WriteMap(writer, m_mapRegionData);
WriteMap(writer, m_mapIsoCode);
}
void CollectorRegionInfo::Merge(CollectorInterface const * collector)
{
CHECK(collector, ());
collector->MergeInto(const_cast<CollectorRegionInfo *>(this));
}
void CollectorRegionInfo::MergeInto(CollectorRegionInfo * collector) const
{
CHECK(collector, ());
collector->m_mapRegionData.insert(std::begin(m_mapRegionData), std::end(m_mapRegionData));
collector->m_mapIsoCode.insert(std::begin(m_mapIsoCode), std::end(m_mapIsoCode));
}
void CollectorRegionInfo::FillRegionData(base::GeoObjectId const & osmId, OsmElement const & el,
RegionData & rd)
{

View file

@ -1,6 +1,7 @@
#pragma once
#include "generator/collector_interface.hpp"
#include "generator/feature_builder.hpp"
#include "platform/platform.hpp"
@ -10,6 +11,7 @@
#include <cstdint>
#include <ostream>
#include <memory>
#include <string>
#include <type_traits>
#include <unordered_map>
@ -20,6 +22,11 @@ class FileWriter;
namespace generator
{
namespace cache
{
class IntermediateDataReader;
} // namespace cache
namespace regions
{
// https://wiki.openstreetmap.org/wiki/Tag:boundary=administrative
@ -120,8 +127,14 @@ public:
// CollectorInterface overrides:
void Collect(OsmElement const & el) override;
std::shared_ptr<CollectorInterface>
Clone(std::shared_ptr<cache::IntermediateDataReader> const & = {}) const override;
void Save() override;
void Merge(CollectorInterface const * collector) override;
void MergeInto(CollectorRegionInfo * collector) const override;
private:
template <typename Sink, typename Map>
void WriteMap(Sink & sink, Map & seq)
@ -136,7 +149,6 @@ private:
void FillRegionData(base::GeoObjectId const & osmId, OsmElement const & el, RegionData & rd);
void FillIsoCode(base::GeoObjectId const & osmId, OsmElement const & el, IsoCode & rd);
std::string m_filename;
MapRegionData m_mapRegionData;
MapIsoCode m_mapIsoCode;
};

View file

@ -1,5 +1,6 @@
#include "generator/restriction_writer.hpp"
#include "generator/intermediate_data.hpp"
#include "generator/intermediate_elements.hpp"
#include "generator/osm_element.hpp"
#include "generator/restriction_collector.hpp"
@ -90,11 +91,19 @@ namespace routing
std::string const RestrictionWriter::kNodeString = "node";
std::string const RestrictionWriter::kWayString = "way";
RestrictionWriter::RestrictionWriter(std::string const & fullPath,
RestrictionWriter::RestrictionWriter(std::string const & filename,
generator::cache::IntermediateDataReader const & cache)
: m_cache(cache)
: generator::CollectorInterface(filename)
, m_cache(cache)
{
Open(fullPath);
m_stream << std::setprecision(20);
}
std::shared_ptr<generator::CollectorInterface>
RestrictionWriter::Clone(std::shared_ptr<generator::cache::IntermediateDataReader> const & cache) const
{
// auto c = cache ? cache : m_cache;
return std::make_shared<RestrictionWriter>(GetFilename(), *cache);
}
//static
@ -109,17 +118,6 @@ RestrictionWriter::ViaType RestrictionWriter::ConvertFromString(std::string cons
UNREACHABLE();
}
void RestrictionWriter::Open(std::string const & fullPath)
{
LOG(LINFO, ("Saving road restrictions in osm id terms to", fullPath));
m_stream.open(fullPath, std::ofstream::out);
if (!IsOpened())
LOG(LINFO, ("Cannot open file", fullPath));
m_stream << std::setprecision(20);
}
bool ValidateOsmRestriction(std::vector<RelationElement::Member> & from,
std::vector<RelationElement::Member> & via,
std::vector<RelationElement::Member> & to,
@ -156,12 +154,6 @@ bool ValidateOsmRestriction(std::vector<RelationElement::Member> & from,
void RestrictionWriter::CollectRelation(RelationElement const & relationElement)
{
if (!IsOpened())
{
LOG(LWARNING, ("Tried to write to a closed restrictions writer"));
return;
}
std::vector<RelationElement::Member> from;
std::vector<RelationElement::Member> via;
std::vector<RelationElement::Member> to;
@ -212,7 +204,27 @@ void RestrictionWriter::CollectRelation(RelationElement const & relationElement)
m_stream << toOsmId << '\n';
}
bool RestrictionWriter::IsOpened() const { return m_stream && m_stream.is_open(); }
void RestrictionWriter::Save()
{
std::ofstream stream;
stream.exceptions(std::fstream::failbit | std::fstream::badbit);
stream.open(GetFilename());
stream << m_stream.str();
}
void RestrictionWriter::Merge(generator::CollectorInterface const * collector)
{
CHECK(collector, ());
collector->MergeInto(const_cast<RestrictionWriter *>(this));
}
void RestrictionWriter::MergeInto(RestrictionWriter * collector) const
{
CHECK(collector, ());
collector->m_stream << m_stream.str();
}
std::string DebugPrint(RestrictionWriter::ViaType const & type)
{

View file

@ -3,11 +3,20 @@
#include "generator/collector_interface.hpp"
#include "generator/intermediate_data.hpp"
#include <fstream>
#include <memory>
#include <sstream>
#include <string>
class RelationElement;
namespace generator
{
namespace cache
{
class IntermediateDataReader;
} // namespace cache
} // namespace generator
namespace routing
{
class RestrictionWriter : public generator::CollectorInterface
@ -25,22 +34,23 @@ public:
static std::string const kNodeString;
static std::string const kWayString;
RestrictionWriter(std::string const & fullPath,
RestrictionWriter(std::string const & filename,
generator::cache::IntermediateDataReader const & cache);
// generator::CollectorInterface overrides:
// @{
std::shared_ptr<CollectorInterface>
Clone(std::shared_ptr<generator::cache::IntermediateDataReader> const & cache = {}) const override;
void CollectRelation(RelationElement const & relationElement) override;
void Save() override {}
// @}
void Save() override;
void Merge(generator::CollectorInterface const * collector) override;
void MergeInto(RestrictionWriter * collector) const override;
static ViaType ConvertFromString(std::string const & str);
private:
void Open(std::string const & fullPath);
bool IsOpened() const;
std::ofstream m_stream;
std::stringstream m_stream;
generator::cache::IntermediateDataReader const & m_cache;
};

View file

@ -1,9 +1,11 @@
#include "generator/road_access_generator.hpp"
#include "generator/feature_builder.hpp"
#include "generator/routing_helpers.hpp"
#include "routing/road_access.hpp"
#include "routing/road_access_serialization.hpp"
#include "routing/routing_helpers.hpp"
#include "indexer/classificator.hpp"
#include "indexer/feature.hpp"
@ -39,68 +41,68 @@ char constexpr kDelim[] = " \t\r\n";
using TagMapping = routing::RoadAccessTagProcessor::TagMapping;
TagMapping const kMotorCarTagMapping = {
{OsmElement::Tag("motorcar", "yes"), RoadAccess::Type::Yes},
{OsmElement::Tag("motorcar", "designated"), RoadAccess::Type::Yes},
{OsmElement::Tag("motorcar", "permissive"), RoadAccess::Type::Yes},
{OsmElement::Tag("motorcar", "no"), RoadAccess::Type::No},
{OsmElement::Tag("motorcar", "private"), RoadAccess::Type::Private},
{OsmElement::Tag("motorcar", "destination"), RoadAccess::Type::Destination},
{OsmElement::Tag("motorcar", "yes"), RoadAccess::Type::Yes},
{OsmElement::Tag("motorcar", "designated"), RoadAccess::Type::Yes},
{OsmElement::Tag("motorcar", "permissive"), RoadAccess::Type::Yes},
{OsmElement::Tag("motorcar", "no"), RoadAccess::Type::No},
{OsmElement::Tag("motorcar", "private"), RoadAccess::Type::Private},
{OsmElement::Tag("motorcar", "destination"), RoadAccess::Type::Destination},
};
TagMapping const kMotorVehicleTagMapping = {
{OsmElement::Tag("motor_vehicle", "yes"), RoadAccess::Type::Yes},
{OsmElement::Tag("motor_vehicle", "designated"), RoadAccess::Type::Yes},
{OsmElement::Tag("motor_vehicle", "permissive"), RoadAccess::Type::Yes},
{OsmElement::Tag("motor_vehicle", "no"), RoadAccess::Type::No},
{OsmElement::Tag("motor_vehicle", "private"), RoadAccess::Type::Private},
{OsmElement::Tag("motor_vehicle", "destination"), RoadAccess::Type::Destination},
{OsmElement::Tag("motor_vehicle", "yes"), RoadAccess::Type::Yes},
{OsmElement::Tag("motor_vehicle", "designated"), RoadAccess::Type::Yes},
{OsmElement::Tag("motor_vehicle", "permissive"), RoadAccess::Type::Yes},
{OsmElement::Tag("motor_vehicle", "no"), RoadAccess::Type::No},
{OsmElement::Tag("motor_vehicle", "private"), RoadAccess::Type::Private},
{OsmElement::Tag("motor_vehicle", "destination"), RoadAccess::Type::Destination},
};
TagMapping const kVehicleTagMapping = {
{OsmElement::Tag("vehicle", "yes"), RoadAccess::Type::Yes},
{OsmElement::Tag("vehicle", "designated"), RoadAccess::Type::Yes},
{OsmElement::Tag("vehicle", "permissive"), RoadAccess::Type::Yes},
{OsmElement::Tag("vehicle", "no"), RoadAccess::Type::No},
{OsmElement::Tag("vehicle", "private"), RoadAccess::Type::Private},
{OsmElement::Tag("vehicle", "destination"), RoadAccess::Type::Destination},
{OsmElement::Tag("vehicle", "yes"), RoadAccess::Type::Yes},
{OsmElement::Tag("vehicle", "designated"), RoadAccess::Type::Yes},
{OsmElement::Tag("vehicle", "permissive"), RoadAccess::Type::Yes},
{OsmElement::Tag("vehicle", "no"), RoadAccess::Type::No},
{OsmElement::Tag("vehicle", "private"), RoadAccess::Type::Private},
{OsmElement::Tag("vehicle", "destination"), RoadAccess::Type::Destination},
};
TagMapping const kCarBarriersTagMapping = {
{OsmElement::Tag("barrier", "block"), RoadAccess::Type::No},
{OsmElement::Tag("barrier", "bollard"), RoadAccess::Type::No},
{OsmElement::Tag("barrier", "cycle_barrier"), RoadAccess::Type::No},
{OsmElement::Tag("barrier", "gate"), RoadAccess::Type::Private},
{OsmElement::Tag("barrier", "lift_gate"), RoadAccess::Type::Private},
// TODO (@gmoryes) The types below should be added.
// {OsmElement::Tag("barrier", "chain"), RoadAccess::Type::No},
// {OsmElement::Tag("barrier", "swing_gate"), RoadAccess::Type::Private}
// {OsmElement::Tag("barrier", "log"), RoadAccess::Type::No},
// {OsmElement::Tag("barrier", "motorcycle_barrier"), RoadAccess::Type::No},
{OsmElement::Tag("barrier", "block"), RoadAccess::Type::No},
{OsmElement::Tag("barrier", "bollard"), RoadAccess::Type::No},
{OsmElement::Tag("barrier", "cycle_barrier"), RoadAccess::Type::No},
{OsmElement::Tag("barrier", "gate"), RoadAccess::Type::Private},
{OsmElement::Tag("barrier", "lift_gate"), RoadAccess::Type::Private},
// TODO (@gmoryes) The types below should be added.
// {OsmElement::Tag("barrier", "chain"), RoadAccess::Type::No},
// {OsmElement::Tag("barrier", "swing_gate"), RoadAccess::Type::Private}
// {OsmElement::Tag("barrier", "log"), RoadAccess::Type::No},
// {OsmElement::Tag("barrier", "motorcycle_barrier"), RoadAccess::Type::No},
};
TagMapping const kPedestrianTagMapping = {
{OsmElement::Tag("foot", "yes"), RoadAccess::Type::Yes},
{OsmElement::Tag("foot", "designated"), RoadAccess::Type::Yes},
{OsmElement::Tag("foot", "permissive"), RoadAccess::Type::Yes},
{OsmElement::Tag("foot", "no"), RoadAccess::Type::No},
{OsmElement::Tag("foot", "private"), RoadAccess::Type::Private},
{OsmElement::Tag("foot", "destination"), RoadAccess::Type::Destination},
{OsmElement::Tag("foot", "yes"), RoadAccess::Type::Yes},
{OsmElement::Tag("foot", "designated"), RoadAccess::Type::Yes},
{OsmElement::Tag("foot", "permissive"), RoadAccess::Type::Yes},
{OsmElement::Tag("foot", "no"), RoadAccess::Type::No},
{OsmElement::Tag("foot", "private"), RoadAccess::Type::Private},
{OsmElement::Tag("foot", "destination"), RoadAccess::Type::Destination},
};
TagMapping const kBicycleTagMapping = {
{OsmElement::Tag("bicycle", "yes"), RoadAccess::Type::Yes},
{OsmElement::Tag("bicycle", "designated"), RoadAccess::Type::Yes},
{OsmElement::Tag("bicycle", "permissive"), RoadAccess::Type::Yes},
{OsmElement::Tag("bicycle", "no"), RoadAccess::Type::No},
{OsmElement::Tag("bicycle", "private"), RoadAccess::Type::Private},
{OsmElement::Tag("bicycle", "destination"), RoadAccess::Type::Destination},
{OsmElement::Tag("bicycle", "yes"), RoadAccess::Type::Yes},
{OsmElement::Tag("bicycle", "designated"), RoadAccess::Type::Yes},
{OsmElement::Tag("bicycle", "permissive"), RoadAccess::Type::Yes},
{OsmElement::Tag("bicycle", "no"), RoadAccess::Type::No},
{OsmElement::Tag("bicycle", "private"), RoadAccess::Type::Private},
{OsmElement::Tag("bicycle", "destination"), RoadAccess::Type::Destination},
};
TagMapping const kBicycleBarriersTagMapping = {
{OsmElement::Tag("barrier", "cycle_barrier"), RoadAccess::Type::No},
{OsmElement::Tag("barrier", "gate"), RoadAccess::Type::Private},
// TODO (@gmoryes) The types below should be added.
// {OsmElement::Tag("barrier", "kissing_gate"), RoadAccess::Type::Private},
{OsmElement::Tag("barrier", "cycle_barrier"), RoadAccess::Type::No},
{OsmElement::Tag("barrier", "gate"), RoadAccess::Type::Private},
// TODO (@gmoryes) The types below should be added.
// {OsmElement::Tag("barrier", "kissing_gate"), RoadAccess::Type::Private},
};
// Allow everything to keep transit section empty. We'll use pedestrian section for
@ -109,24 +111,24 @@ TagMapping const kBicycleBarriersTagMapping = {
TagMapping const kTransitTagMapping = {};
TagMapping const kDefaultTagMapping = {
{OsmElement::Tag("access", "yes"), RoadAccess::Type::Yes},
{OsmElement::Tag("access", "permissive"), RoadAccess::Type::Yes},
{OsmElement::Tag("access", "no"), RoadAccess::Type::No},
{OsmElement::Tag("access", "private"), RoadAccess::Type::Private},
{OsmElement::Tag("access", "destination"), RoadAccess::Type::Destination},
{OsmElement::Tag("access", "yes"), RoadAccess::Type::Yes},
{OsmElement::Tag("access", "permissive"), RoadAccess::Type::Yes},
{OsmElement::Tag("access", "no"), RoadAccess::Type::No},
{OsmElement::Tag("access", "private"), RoadAccess::Type::Private},
{OsmElement::Tag("access", "destination"), RoadAccess::Type::Destination},
};
set<OsmElement::Tag> const kHighwaysWhereIgnorePrivateAccessForCar = {
{OsmElement::Tag("highway", "motorway")},
{OsmElement::Tag("highway", "motorway_link")},
{OsmElement::Tag("highway", "primary")},
{OsmElement::Tag("highway", "primary_link")},
{OsmElement::Tag("highway", "secondary")},
{OsmElement::Tag("highway", "secondary_link")},
{OsmElement::Tag("highway", "tertiary")},
{OsmElement::Tag("highway", "tertiary_link")},
{OsmElement::Tag("highway", "trunk")},
{OsmElement::Tag("highway", "trunk_link")}
{OsmElement::Tag("highway", "motorway")},
{OsmElement::Tag("highway", "motorway_link")},
{OsmElement::Tag("highway", "primary")},
{OsmElement::Tag("highway", "primary_link")},
{OsmElement::Tag("highway", "secondary")},
{OsmElement::Tag("highway", "secondary_link")},
{OsmElement::Tag("highway", "tertiary")},
{OsmElement::Tag("highway", "tertiary_link")},
{OsmElement::Tag("highway", "trunk")},
{OsmElement::Tag("highway", "trunk_link")}
};
set<OsmElement::Tag> const kHighwaysWhereIgnorePrivateAccessEmpty = {};
@ -147,7 +149,7 @@ bool ParseRoadAccess(string const & roadAccessPath,
unordered_map<RoadPoint, RoadAccess::Type, RoadPoint::Hash> pointType[static_cast<size_t>(VehicleType::Count)];
auto addFeature = [&](uint32_t featureId, VehicleType vehicleType,
RoadAccess::Type roadAccessType, uint64_t osmId) {
RoadAccess::Type roadAccessType, uint64_t osmId) {
auto & m = featureType[static_cast<size_t>(vehicleType)];
auto const emplaceRes = m.emplace(featureId, roadAccessType);
if (!emplaceRes.second && emplaceRes.first->second != roadAccessType)
@ -158,7 +160,7 @@ bool ParseRoadAccess(string const & roadAccessPath,
};
auto addPoint = [&](RoadPoint const & point, VehicleType vehicleType,
RoadAccess::Type roadAccessType) {
RoadAccess::Type roadAccessType) {
auto & m = pointType[static_cast<size_t>(vehicleType)];
auto const emplaceRes = m.emplace(point, roadAccessType);
if (!emplaceRes.second && emplaceRes.first->second != roadAccessType)
@ -287,7 +289,7 @@ RoadAccessTagProcessor::RoadAccessTagProcessor(VehicleType vehicleType)
}
}
void RoadAccessTagProcessor::Process(OsmElement const & elem, ofstream & oss)
void RoadAccessTagProcessor::Process(FeatureBuilder const & fb, OsmElement const & elem)
{
// We will process all nodes before ways because of o5m format:
// all nodes are first, then all ways, then all relations.
@ -295,33 +297,58 @@ void RoadAccessTagProcessor::Process(OsmElement const & elem, ofstream & oss)
{
RoadAccess::Type accessType = GetAccessType(elem);
if (accessType != RoadAccess::Type::Yes)
m_barriers[elem.m_id] = accessType;
m_barriers.emplace(elem.m_id, accessType);
return;
}
if (elem.m_type != OsmElement::EntityType::Way)
return;
// All feature tags.
auto const accessType = GetAccessType(elem);
if (accessType != RoadAccess::Type::Yes)
oss << ToString(m_vehicleType) << " " << ToString(accessType) << " " << elem.m_id << " "
<< 0 /* wildcard segment Idx */ << endl;
m_wayToAccess.emplace(elem.m_id, accessType);
if (!routing::IsRoad(fb.GetTypes()))
return;
m_roads.emplace(elem.m_id, elem.m_nodes);
}
void RoadAccessTagProcessor::Write(std::stringstream & stream)
{
// All feature tags.
for (auto const & i : m_wayToAccess)
{
stream << ToString(m_vehicleType) << " " << ToString(i.second) << " " << i.first << " "
<< 0 /* wildcard segment Idx */ << endl;
}
// Barrier tags.
for (size_t pointIdx = 0; pointIdx < elem.m_nodes.size(); ++pointIdx)
for (auto const & i : m_roads)
{
auto const it = m_barriers.find(elem.m_nodes[pointIdx]);
if (it == m_barriers.cend())
continue;
for (size_t pointIdx = 0; pointIdx < i.second.size(); ++pointIdx)
{
auto const it = m_barriers.find(i.second[pointIdx]);
if (it == m_barriers.cend())
continue;
RoadAccess::Type const roadAccessType = it->second;
// idx == 0 used as wildcard segment Idx, for nodes we store |pointIdx + 1| instead of |pointIdx|.
oss << ToString(m_vehicleType) << " " << ToString(roadAccessType) << " " << elem.m_id << " "
<< pointIdx + 1 << endl;
RoadAccess::Type const roadAccessType = it->second;
// idx == 0 used as wildcard segment Idx, for nodes we store |pointIdx + 1| instead of |pointIdx|.
stream << ToString(m_vehicleType) << " " << ToString(roadAccessType) << " " << i.first << " "
<< pointIdx + 1 << endl;
}
}
}
void RoadAccessTagProcessor::Merge(RoadAccessTagProcessor const & other)
{
CHECK_EQUAL(m_vehicleType, other.m_vehicleType, ());
m_barriers.insert(std::begin(other.m_barriers), std::end(other.m_barriers));
m_wayToAccess.insert(std::begin(other.m_wayToAccess), std::end(other.m_wayToAccess));
m_roads.insert(std::begin(other.m_roads), std::end(other.m_roads));
}
bool RoadAccessTagProcessor::ShouldIgnoreBarrierWithoutAccess(OsmElement const & osmElement) const
{
CHECK(m_hwIgnoreBarriersWithoutAccess, ());
@ -362,37 +389,54 @@ RoadAccess::Type RoadAccessTagProcessor::GetAccessType(OsmElement const & elem)
}
// RoadAccessWriter ------------------------------------------------------------
RoadAccessWriter::RoadAccessWriter(string const & filePath)
RoadAccessWriter::RoadAccessWriter(string const & filename)
: generator::CollectorInterface(filename)
{
for (size_t i = 0; i < static_cast<size_t>(VehicleType::Count); ++i)
m_tagProcessors.emplace_back(static_cast<VehicleType>(i));
Open(filePath);
}
void RoadAccessWriter::Open(string const & filePath)
std::shared_ptr<generator::CollectorInterface>
RoadAccessWriter::Clone(std::shared_ptr<generator::cache::IntermediateDataReader> const &) const
{
LOG(LINFO,
("Saving information about barriers and road access classes in osm id terms to", filePath));
m_stream.open(filePath, ofstream::out);
if (!IsOpened())
LOG(LINFO, ("Cannot open file", filePath));
return std::make_shared<RoadAccessWriter>(GetFilename());
}
void RoadAccessWriter::CollectFeature(FeatureBuilder const &, OsmElement const & elem)
void RoadAccessWriter::CollectFeature(FeatureBuilder const & fb, OsmElement const & elem)
{
if (!IsOpened())
{
LOG(LWARNING, ("Tried to write to a closed barriers writer"));
return;
}
for (auto & p : m_tagProcessors)
p.Process(elem, m_stream);
p.Process(fb, elem);
}
bool RoadAccessWriter::IsOpened() const { return m_stream && m_stream.is_open(); }
void RoadAccessWriter::Save()
{
std::stringstream stream;
for (auto & p : m_tagProcessors)
p.Write(stream);
std::ofstream out;
out.exceptions(std::fstream::failbit | std::fstream::badbit);
out.open(GetFilename());
out << stream.str();
}
void RoadAccessWriter::Merge(generator::CollectorInterface const * collector)
{
CHECK(collector, ());
collector->MergeInto(const_cast<RoadAccessWriter *>(this));
}
void RoadAccessWriter::MergeInto(RoadAccessWriter * collector) const
{
CHECK(collector, ());
auto & otherProcessors = collector->m_tagProcessors;
CHECK_EQUAL(m_tagProcessors.size(), otherProcessors.size(), ());
for (size_t i = 0; i < otherProcessors.size(); ++i)
otherProcessors[i].Merge(m_tagProcessors[i]);
}
// RoadAccessCollector ----------------------------------------------------------
RoadAccessCollector::RoadAccessCollector(string const & dataFilePath, string const & roadAccessPath,

View file

@ -1,6 +1,7 @@
#pragma once
#include "generator/collector_interface.hpp"
#include "generator/feature_builder.hpp"
#include "generator/intermediate_elements.hpp"
#include "generator/osm_element.hpp"
@ -13,12 +14,21 @@
#include <map>
#include <ostream>
#include <set>
#include <sstream>
#include <string>
#include <vector>
struct OsmElement;
class FeatureParams;
namespace generator
{
namespace cache
{
class IntermediateDataReader;
} // namespace cache
} // namespace generator
// The road accessibility information is collected in the same manner
// as the restrictions are.
// See generator/restriction_generator.hpp for details.
@ -31,7 +41,9 @@ public:
explicit RoadAccessTagProcessor(VehicleType vehicleType);
void Process(OsmElement const & elem, std::ofstream & oss);
void Process(feature::FeatureBuilder const & fb, OsmElement const & elem);
void Write(std::stringstream & stream);
void Merge(RoadAccessTagProcessor const & roadAccessTagProcessor);
private:
bool ShouldIgnoreBarrierWithoutAccess(OsmElement const & osmElement) const;
@ -49,22 +61,26 @@ private:
// because we almost always do not need to add penalty for passes through such nodes.
std::set<OsmElement::Tag> const * m_hwIgnoreBarriersWithoutAccess;
std::map<uint64_t, RoadAccess::Type> m_barriers;
std::unordered_map<uint64_t, RoadAccess::Type> m_barriers;
std::unordered_map<uint64_t, RoadAccess::Type> m_wayToAccess;
std::unordered_map<uint64_t, std::vector<uint64_t>> m_roads;
};
class RoadAccessWriter : public generator::CollectorInterface
{
public:
RoadAccessWriter(std::string const & filePath);
RoadAccessWriter(std::string const & filename);
// CollectorInterface overrides:
void CollectFeature(feature::FeatureBuilder const &, OsmElement const & elem) override;
void Save() override {}
std::shared_ptr<CollectorInterface>
Clone(std::shared_ptr<generator::cache::IntermediateDataReader> const & = {}) const override;
void CollectFeature(feature::FeatureBuilder const & fb, OsmElement const & elem) override;
void Save() override;
void Merge(generator::CollectorInterface const * collector) override;
void MergeInto(RoadAccessWriter * collector) const override;
private:
void Open(std::string const & filePath);
bool IsOpened() const;
std::ofstream m_stream;
std::vector<RoadAccessTagProcessor> m_tagProcessors;
};